Compare commits

...

6 Commits

2 changed files with 345 additions and 243 deletions

View File

@ -1,68 +1,40 @@
let isRead = true; var isRead = true;
let isChoose = true; var isChoose = true;
let reader = window.speechSynthesis; // 全局定义朗读者,以便朗读和暂停 const reader = window.speechSynthesis; // 全局定义朗读者,以便朗读和暂停
let current_position = 0; // 朗读文本的当前位置 //获取字符
let original_position = 0; // 朗读文本的初始位置
let to_speak = ""; // 朗读的初始内容
function getWord() { function getWord() {
return window.getSelection ? window.getSelection() : document.selection.createRange().text; return window.getSelection
? window.getSelection()
: document.selection.createRange().text;
} }
//填充字符
function fillInWord() { function fillingWord() {
let word = getWord(); let word = getWord();
if (isRead) read(word); if (isRead) read(word);
if (!isChoose) return; if (!isChoose) return;
const element = document.getElementById("selected-words"); let element = document.getElementById("selected-words");
element.value = element.value + " " + word; element.value = element.value + " " + word;
} }
document
document.getElementById("text-content").addEventListener("click", fillInWord, false); .getElementById("text-content")
.addEventListener("click", fillingWord, false);
function makeUtterance(str, rate) { //朗读单词
let msg = new SpeechSynthesisUtterance(str); //@word 要朗读的单词
msg.rate = rate; function read(word) {
msg.lang = "en-US"; // TODO: add language options menu //关闭当前正在读的单词
msg.onboundary = ev => { reader.cancel();
if (ev.name == "word") { //创建新的朗读任务
current_position = ev.charIndex; let msg = new SpeechSynthesisUtterance(word);
} reader.speak(msg);
}
return msg;
} }
//关闭正在读的单词
const sliderValue = document.getElementById("rangeValue"); // 显示值
const inputSlider = document.getElementById("rangeComponent"); // 滑块元素
inputSlider.oninput = () => {
let value = inputSlider.value; // 获取滑块的值
sliderValue.textContent = value + '×';
if (!reader.speaking) return;
reader.cancel();
let msg = makeUtterance(to_speak.substring(original_position + current_position), value);
original_position = original_position + current_position;
current_position = 0;
reader.speak(msg);
};
function read(s) {
to_speak = s.toString();
original_position = 0;
current_position = 0;
let msg = makeUtterance(to_speak, inputSlider.value);
reader.speak(msg);
}
function onReadClick() { function onReadClick() {
isRead = !isRead; let isRead = !isRead;
if (!isRead) { if (!isRead) {
reader.cancel();
}
}
function onChooseClick() {
isChoose = !isChoose;
}
function stopRead() {
reader.cancel(); reader.cancel();
} }
}
//取消当前选择
function onChooseClick() {
isChoose = !isChoose;
}

View File

@ -1,217 +1,347 @@
<!DOCTYPE html> <!DOCTYPE html>
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="UTF-8"> <meta charset="UTF-8" />
<meta name="viewport" <meta
content="width=device-width, initial-scale=1.0, minimum-scale=0.5, maximum-scale=3.0, user-scalable=yes"/> name="viewport"
<meta name="format-detection" content="telephone=no"/> content="width=device-width, initial-scale=1.0, minimum-scale=0.5, maximum-scale=3.0, user-scalable=yes"
/>
<meta name="format-detection" content="telephone=no" />
{{ yml['header'] | safe }} {{ yml['header'] | safe }} {% if yml['css']['item'] %} {% for css in
{% if yml['css']['item'] %} yml['css']['item'] %}
{% for css in yml['css']['item'] %} <link href="{{ css }}" rel="stylesheet" />
<link href="{{ css }}" rel="stylesheet"> {% endfor %} {% endif %} {% if yml['js']['head'] %} {% for js in
{% endfor %} yml['js']['head'] %}
{% endif %} <script src="{{ js }}"></script>
{% if yml['js']['head'] %} {% endfor %} {% endif %}
{% for js in yml['js']['head'] %}
<script src="{{ js }}"></script>
{% endfor %}
{% endif %}
<title>EnglishPal Study Room for {{ username }}</title> <title>EnglishPal Study Room for {{ username }}</title>
<style> <style>
.shaking { .shaking {
animation: shakes 1600ms ease-in-out; animation: shakes 1600ms ease-in-out;
} }
@keyframes shakes { @keyframes shakes {
10%, 90% { transform: translate3d(-1px, 0, 0); } 10%,
20%, 50% { transform: translate3d(+2px, 0, 0); } 90% {
30%, 70% { transform: translate3d(-4px, 0, 0); } transform: translate3d(-1px, 0, 0);
40%, 60% { transform: translate3d(+4px, 0, 0); }
50% { transform: translate3d(-4px, 0, 0); }
} }
20%,
50% {
transform: translate3d(+2px, 0, 0);
}
30%,
70% {
transform: translate3d(-4px, 0, 0);
}
40%,
60% {
transform: translate3d(+4px, 0, 0);
}
50% {
transform: translate3d(-4px, 0, 0);
}
}
</style> </style>
</head> </head>
<body> <body>
<div class="container-fluid"> <div class="container-fluid">
<p><b>English Pal for <font id="username" color="red">{{ username }}</font></b> <p>
<b
>English Pal for
<font id="username" color="red">{{ username }}</font></b
>
{% if username == admin_name %} {% if username == admin_name %}
<a class="btn btn-secondary" href="/admin" role="button" onclick="stopRead()">管理</a> <a
class="btn btn-secondary"
href="/admin"
role="button"
onclick="stopRead()"
>管理</a
>
{% endif %} {% endif %}
<a id="quit" class="btn btn-secondary" href="/logout" role="button" onclick="stopRead()">退出</a> <a
<a class="btn btn-secondary" href="/reset" role="button" onclick="stopRead()">重设密码</a> id="quit"
class="btn btn-secondary"
href="/logout"
role="button"
onclick="stopRead()"
>退出</a
>
<a
class="btn btn-secondary"
href="/reset"
role="button"
onclick="stopRead()"
>重设密码</a
>
</p>
{# {% for message in flashed_messages %}#} {#
根据user_service.userpage,取消了参数flashed_messages因此注释了这段代码
#} {#
<div class="alert alert-warning" role="alert">
Congratulations! {{ message }}
</div>
#} {# {% endfor %}#}
</p> <button
{# {% for message in flashed_messages %}#} {# 根据user_service.userpage,取消了参数flashed_messages因此注释了这段代码 #} class="btn btn-success"
{# <div class="alert alert-warning" role="alert">Congratulations! {{ message }}</div>#} id="load_next_article"
{# {% endfor %}#} onclick="load_next_article()"
>
下一篇 Next Article
</button>
<button
class="btn btn-success"
id="load_pre_article"
onclick="load_pre_article()"
>
上一篇 Previous Article
</button>
<button class="btn btn-success" id="load_next_article" onclick="load_next_article()"> 下一篇 Next Article </button> <p><b>阅读文章并回答问题</b></p>
<button class="btn btn-success" id="load_pre_article" onclick="load_pre_article()" > 上一篇 Previous Article </button> <div id="text-content">
<p><b>阅读文章并回答问题</b></p>
<div id="text-content">
<div id="found"> <div id="found">
<div class="alert alert-success" role="alert">According to your word list, your level is <span class="badge bg-success" id="user-level">{{ today_article["user_level"] }}</span> and we have chosen an article with a difficulty level of <span class="badge bg-success" id="text_level">{{ today_article["text_level"] }}</span> for you.</div> <div class="alert alert-success" role="alert">
<p class="text-muted" id="date">Article added on: {{ today_article["date"] }}</p><br/> According to your word list, your level is
<div class="p-3 mb-2 bg-light text-dark"><br/> <span class="badge bg-success" id="user-level"
<p class="display-5" id="article_title">{{ today_article["article_title"] }}</p><br/> >{{ today_article["user_level"] }}</span
<p class="lead"><font id="article" size=2>{{ today_article["article_body"] }}</font></p><br/> >
<p><small class="text-muted" id="source">{{ today_article['source'] }}</small></p><br/> and we have chosen an article with a difficulty level of
<p><b id="question">{{ today_article['question'] }}</b></p><br/> <span class="badge bg-success" id="text_level"
<script type="text/javascript"> >{{ today_article["text_level"] }}</span
function toggle_visibility(id) { {# https://css-tricks.com/snippets/javascript/showhide-element/#} >
const e = document.getElementById(id); for you.
if(e.style.display === 'block') </div>
e.style.display = 'none'; <p class="text-muted" id="date">
else Article added on: {{ today_article["date"] }}
e.style.display = 'block'; </p>
} <br />
</script> <div class="p-3 mb-2 bg-light text-dark">
<button onclick="toggle_visibility('answer');">ANSWER</button> <br />
<div id="answer" style="display:none;">{{ today_article['answer'] }}</div><br/> <p class="display-5" id="article_title">
{{ today_article["article_title"] }}
</p>
<br />
<p class="lead">
<font id="article" size="2"
>{{ today_article["article_body"] }}</font
>
</p>
<br />
<p>
<small class="text-muted" id="source"
>{{ today_article['source'] }}</small
>
</p>
<br />
<p><b id="question">{{ today_article['question'] }}</b></p>
<br />
<script type="text/javascript">
function toggle_visibility(id) { {# https://css-tricks.com/snippets/javascript/showhide-element/#}
const e = document.getElementById(id);
if(e.style.display === 'block')
e.style.display = 'none';
else
e.style.display = 'block';
}
</script>
<button onclick="toggle_visibility('answer');">ANSWER</button>
<div id="answer" style="display: none">
{{ today_article['answer'] }}
</div> </div>
<br />
</div>
</div> </div>
<div class="alert alert-success" role="alert" id="not_found" style="display:none;"> <div
<p class="text-muted"><span class="badge bg-success">Notes:</span><br>No article is currently available for you. You can try again a few times or mark new words in the passage to improve your level.</p> class="alert alert-success"
role="alert"
id="not_found"
style="display: none"
>
<p class="text-muted">
<span class="badge bg-success">Notes:</span><br />No article is
currently available for you. You can try again a few times or mark
new words in the passage to improve your level.
</p>
</div> </div>
<div class="alert alert-success" role="alert" id="read_all" style="display:none;"> <div
<p class="text-muted"><span class="badge bg-success">Notes:</span><br>You've read all the articles.</p> class="alert alert-success"
role="alert"
id="read_all"
style="display: none"
>
<p class="text-muted">
<span class="badge bg-success">Notes:</span><br />You've read all
the articles.
</p>
</div> </div>
</div> </div>
<input type="checkbox" onclick="toggleHighlighting()" checked/>生词高亮 <input type="checkbox" onclick="toggleHighlighting()" checked />生词高亮
<input type="checkbox" onclick="onReadClick()" checked/>大声朗读 <input type="checkbox" onclick="onReadClick()" checked />大声朗读
<input type="checkbox" onclick="onChooseClick()" checked/>划词入库 <input type="checkbox" onclick="onChooseClick()" checked />划词入库
<div class="range"> <div class="range">
<div class="field"> <div class="field">
<div class="sliderValue"> <div class="sliderValue">
<span id="rangeValue">1×</span> <span id="rangeValue">1×</span>
</div> </div>
<input type="range" id="rangeComponent" min="0.5" max="2" value="1" step="0.25"/> <input
type="range"
id="rangeComponent"
min="0.5"
max="2"
value="1"
step="0.25"
/>
</div> </div>
</div> </div>
<p><b>收集生词吧</b> (可以在正文中划词,也可以复制黏贴)</p> <p><b>收集生词吧</b> (可以在正文中划词,也可以复制黏贴)</p>
<form method="post" action="/{{ username }}/userpage"> <form method="post" action="/{{ username }}/userpage">
<textarea name="content" id="selected-words" rows="10" cols="120"></textarea><br/> <textarea
<input type="submit" value="把生词加入我的生词库"/> name="content"
<input type="reset" value="清除"/> id="selected-words"
</form> rows="10"
{% if session.get['thisWord'] %} cols="120"
<script type="text/javascript"> ></textarea
//point to the anchor in the page whose id is aaa if it exists ><br />
window.onload = function () { <input type="submit" value="把生词加入我的生词库" />
var element = document.getElementsByName("aaa"); <input type="reset" value="清除" />
if (element != null) </form>
document.getElementsByName("aaa")[0].scrollIntoView(true); {% if session.get['thisWord'] %}
} <script type="text/javascript">
</script> //point to the anchor in the page whose id is aaa if it exists
{% endif %} window.onload = function () {
var element = document.getElementsByName("aaa");
{% if d_len > 0 %} if (element != null)
<p> document.getElementsByName("aaa")[0].scrollIntoView(true);
<b>我的生词簿</b> };
<label for="move_dynamiclly"> </script>
<input type="checkbox" name="move_dynamiclly" id="move_dynamiclly" checked> {% endif %} {% if d_len > 0 %}
允许动态调整顺序 <p>
</label> <b>我的生词簿</b>
<label for="move_dynamiclly">
<input
type="checkbox"
name="move_dynamiclly"
id="move_dynamiclly"
checked
/>
允许动态调整顺序
</label>
</p>
<a name="aaa"></a>
<div class="word-container">
{% for x in lst3 %} {% set word = x[0] %} {% set freq = x[1] %} {% if
session.get('thisWord') == x[0] and session.get('time') == 1 %} {% endif
%}
<p id="p_{{ word }}" class="new-word">
<a
id="word_{{ word }}"
class="btn btn-light"
href="http://youdao.com/w/eng/{{ word }}/#keyfrom=dict2.index"
role="button"
>{{ word }}</a
>
( <a id="freq_{{ word }}" title="{{ word }}">{{ freq }}</a> )
<a
class="btn btn-success"
onclick="familiar('{{ word }}')"
role="button"
>熟悉</a
>
<a
class="btn btn-warning"
onclick="unfamiliar('{{ word }}')"
role="button"
>不熟悉</a
>
<a
class="btn btn-danger"
onclick="delete_word('{{ word }}')"
role="button"
>删除</a
>
</p> </p>
<a name="aaa"></a> {% endfor %}
<div class="word-container"> </div>
{% for x in lst3 %} <input id="selected-words2" type="hidden" value="{{ words }}" />
{% set word = x[0] %} {% endif %}
{% set freq = x[1] %} </div>
{% if session.get('thisWord') == x[0] and session.get('time') == 1 %} {{ yml['footer'] | safe }} {% if yml['js']['bottom'] %} {% for js in
{% endif %} yml['js']['bottom'] %}
<p id='p_{{ word }}' class="new-word" > <script src="{{ js }}"></script>
<a id="word_{{ word }}" class="btn btn-light" href='http://youdao.com/w/eng/{{ word }}/#keyfrom=dict2.index' {% endfor %} {% endif %}
role="button">{{ word }}</a> <script type="text/javascript">
( <a id="freq_{{ word }}" title="{{ word }}">{{ freq }}</a> ) function load_next_article() {
<a class="btn btn-success" onclick="familiar('{{ word }}')" role="button">熟悉</a>
<a class="btn btn-warning" onclick="unfamiliar('{{ word }}')" role="button">不熟悉</a>
<a class="btn btn-danger" onclick="delete_word('{{ word }}')" role="button">删除</a>
</p>
{% endfor %}
</div>
<input id="selected-words2" type="hidden" value="{{ words }}">
{% endif %}
</div>
{{ yml['footer'] | safe }}
{% if yml['js']['bottom'] %}
{% for js in yml['js']['bottom'] %}
<script src="{{ js }}"></script>
{% endfor %}
{% endif %}
<script type="text/javascript">
function load_next_article(){
$.ajax({ $.ajax({
url: '/get_next_article/{{username}}', url: "/get_next_article/{{username}}",
dataType: 'json', dataType: "json",
success: function(data) { success: function (data) {
// 更新页面内容 // 更新页面内容
if(data['today_article']){ if (data["today_article"]) {
update(data['today_article']); update(data["today_article"]);
check_pre(data['visited_articles']); check_pre(data["visited_articles"]);
check_next(data['result_of_generate_article']); check_next(data["result_of_generate_article"]);
}
} }
},
}); });
} }
function load_pre_article(){ function load_pre_article() {
$.ajax({ $.ajax({
url: '/get_pre_article/{{username}}', url: "/get_pre_article/{{username}}",
dataType: 'json', dataType: "json",
success: function(data) { success: function (data) {
// 更新页面内容 // 更新页面内容
if(data['today_article']){ if (data["today_article"]) {
update(data['today_article']); update(data["today_article"]);
check_pre(data['visited_articles']); check_pre(data["visited_articles"]);
}
} }
},
}); });
} }
function update(today_article){ function update(today_article) {
$('#user-level').html(today_article['user_level']); $("#user-level").html(today_article["user_level"]);
$('#text_level').html(today_article["text_level"]); $("#text_level").html(today_article["text_level"]);
$('#date').html('Article added on: '+today_article["date"]); $("#date").html("Article added on: " + today_article["date"]);
$('#article_title').html(today_article["article_title"]); $("#article_title").html(today_article["article_title"]);
$('#article').html(today_article["article_body"]); $("#article").html(today_article["article_body"]);
$('#source').html(today_article['source']); $("#source").html(today_article["source"]);
$('#question').html(today_article["question"]); $("#question").html(today_article["question"]);
$('#answer').html(today_article["answer"]); $("#answer").html(today_article["answer"]);
} }
<!-- 检查是否存在上一篇或下一篇,不存在则对应按钮隐藏--> <!-- 检查是否存在上一篇或下一篇,不存在则对应按钮隐藏-->
function check_pre(visited_articles){ function check_pre(visited_articles) {
if((visited_articles=='')||(visited_articles['index']<=0)){ if (visited_articles == "" || visited_articles["index"] <= 0) {
$('#load_pre_article').hide(); $("#load_pre_article").hide();
}else{ } else {
$('#load_pre_article').show(); $("#load_pre_article").show();
} }
} }
function check_next(result_of_generate_article){ function check_next(result_of_generate_article) {
if(result_of_generate_article == "found"){ if (result_of_generate_article == "found") {
$('#found').show();$('#not_found').hide(); $("#found").show();
$('#read_all').hide(); $("#not_found").hide();
}else if(result_of_generate_article == "not found"){ $("#read_all").hide();
$('#found').hide(); } else if (result_of_generate_article == "not found") {
$('#not_found').show(); $("#found").hide();
$('#read_all').hide(); $("#not_found").show();
}else{ $("#read_all").hide();
$('#found').hide(); } else {
$('#not_found').hide(); $("#found").hide();
$('#read_all').show(); $("#not_found").hide();
$("#read_all").show();
} }
} }
</script> </script>
</body> </body>
<style> <style>
mark { mark {
color: #{{ yml['highlight']['color'] }}; color: #{{ yml['highlight']['color'] }};
background-color: rgba(0, 0, 0, 0); background-color: rgba(0, 0, 0, 0);
} }
</style> </style>
</html> </html>