上次发的文章《评论自动填写个人信息》所记方法一直觉得很好用,奈何个别网站用不了,其中就包括自己的博客。没办法换了个折中的法子,用输入法自带的自定义短语功能来加快填写速度。但这个方法也有不少问题,一是昵称、邮箱、网址这三项不少博客不能全部自动填写,总有那么一两项输入法调不出自定义的短语,这种情况在移动端上最为明显。二是操作步骤还是有点多,效率并不怎么高。
今天不死心,鼓捣了好长时间,终于把这个书签脚本自动填写评论信息的方法优化了一下,顺便也把在用的这个主题评论提交方面,自身存在的问题处理掉了,现将相关解决方法记录如下:
书签脚本不能在我的博客发挥作用的解释:
自动填写评论信息书签脚本,是通过直接操作 DOM 将评论表单中的昵称、邮箱和网址字段填写的,在 Vue.js 应用中,表单输入通常使用 v-model 进行数据绑定。当直接通过 JavaScript 修改输入元素的 value 属性时,Vue.js 无法检测到这些更改,因为它依赖于 input 事件来同步数据。
为了让 Vue.js 识别到通过脚本填写的内容,需要在修改输入值后,手动触发 input 事件。这将通知 Vue.js 数据模型已经更新,从而正确地同步数据。
新的评论自动填写个人信息书签脚本解决了这个问题,现记录如下:(使用方法见上篇)
javascript:(function()%7Bvar%20lauthor=%5B%22%23author%22,%22input%5Bname='comname'%5D%22,%22%23inpName%22,%22input%5Bname='author'%5D%22,%22%23ds-dialog-name%22%5D,lmail=%5B%22%23mail%22,%22%23email%22,%22input%5Bname='commail'%5D%22,%22%23inpEmail%22,%22input%5Bname='email'%5D%22,%22%23ds-dialog-email%22%5D,lurl=%5B%22%23url%22,%22input%5Bname='comurl'%5D%22,%22%23inpHomePage%22,%22%23ds-dialog-url%22,%22input%5Bname='url'%5D%22%5D;for(var%20i=0;i%3Clauthor.length;i++)%7Bvar%20author=document.querySelector(lauthor%5Bi%5D);if(author!=null)%7Bauthor.value='沉沦';var%20event=document.createEvent('Event');event.initEvent('input',true,true);author.dispatchEvent(event);break;%7D%7Dfor(var%20j=0;j%3Clmail.length;j++)%7Bvar%20mail=document.querySelector(lmail%5Bj%5D);if(mail!=null)%7Bmail.value='admin@kkn.me';var%20event=document.createEvent('Event');event.initEvent('input',true,true);mail.dispatchEvent(event);break;%7D%7Dfor(var%20k=0;k%3Clurl.length;k++)%7Bvar%20url=document.querySelector(lurl%5Bk%5D);if(url!=null)%7Burl.value='https://kkn.me';var%20event=document.createEvent('Event');event.initEvent('input',true,true);url.dispatchEvent(event);break;%7D%7D%7D)();
现用主题使用此书签脚本存在的问题:
当读者是首次发表评论时,使用书签代码自动填写昵称、邮箱和网址后,昵称和网址会被自动清空。这是因为在这个主题的 CommentForm 组件中,有一段代码会在邮箱字段改变后,自动获取用户信息并更新 this.form.author 和 this.form.url。如果获取到的 author 和 url 为空,就会将书签脚本填写的昵称和网址清空。为了避免昵称和网址被清空,需要在 author 和 url 有值时才更新 this.form.author 和 this.form.url。
另外这个主题本身还存在一个比较明显的问题,就是评论提交时不会正确验证必填项,造成即使你不填写必填的邮箱和昵称信息,提交评论后前端仍然反馈错误的“评论提交成功”提示。
现将这个主题 modules.js 中修改后的整个评论表单代码记录如下:(修改后的代码,可以确保昵称和网址不会在使用上述新的书签脚本后,自动填写的信息被清空。并且修复了评论提交时不会正确验证必填项,避免了提示成功但实际提交失败的情况。)
this.CommentForm = {
name: 'comment-form',
template: `
<form method="post" action id="comment_form" :class="{ 'small-size': reply.id }" @submit="e => e.preventDefault()">
<div class="form-group w-100">
<div v-show="!userId" slot="content-top" class="user-info flex-center w-100">
<input v-for="(item, index) in inputs" :key="item.key" class="form-input" type="text" v-model="form[item.bind.name]" v-bind="item.bind" :disabled="sending" @input="() => item.event && item.event()" />
</div>
<div class="d-flex">
<figure class="user-avatar s-rounded">
<img class="s-rounded" :src="avatar" :alt="form.author" />
</figure>
<div class="d-flex flex-wrap w-100">
<editor class="w-100" ref="editor" @submit="submit" v-bind="info.editor" />
</div>
</div>
</div>
</form>
`,
components: { Editor: that.Editor },
props: {
info: {
type: Object, default: () => ({}),
},
reply: {
type: Object, default: () => ({}),
},
},
data() {
return {
sending: false,
inputs: [
{
bind: { name: 'email', placeholder: '邮件', required: true, autocomplete: 'email' },
event: $h.debounce(() => {
$h.visitor(this.form.email, ({ author, avatar, url }) => {
this.avatar = avatar;
// 仅在返回的 author 有值且当前 form.author 为空时更新
if (!this.form.author && author) {
this.form.author = author;
}
// 仅在返回的 url 有值且当前 form.url 为空时更新
if (!this.form.url && url) {
this.form.url = url;
}
});
}, 600),
},
{ bind: { name: 'author', placeholder: '昵称', required: true, autocomplete: 'name' } },
{ bind: { name: 'url', placeholder: '网址', autocomplete: 'url' } },
],
avatar: '',
userId: '',
form: {
comment_post_ID: null, comment_parent: null, author: '', email: '', url: '', comment: '',
},
};
},
watch: {
info: {
deep: true,
immediate: true,
handler({ post_id, visitor }) {
this.userId = visitor.user_id;
if (visitor.email !== this.form.email) {
$h.visitor(visitor.email, ({ avatar }) => {
this.avatar = avatar;
});
}
// 修改 this.form 赋值方式,避免覆盖已有的 author 和 url
this.form = {
...this.form,
comment_post_ID: post_id,
email: visitor.email || this.form.email,
author: this.form.author || visitor.author || '',
url: this.form.url || visitor.url || '',
};
},
},
reply: {
deep: true,
immediate: true,
handler(data) {
this.form.comment_parent = data.id;
},
},
},
methods: {
submit({ content, upload }) {
if (!content) {
this.$toast({ type: 'error', message: '请输入评论内容' });
return;
}
// 验证必填字段
if (!this.userId && (!this.form.author || !this.form.email)) {
this.$toast({ type: 'error', message: '请填写昵称和邮箱' });
return;
}
this.form.comment = content;
this.sending = true;
$h.ajax({
query: { action: 'submit_comment' },
data: this.form,
method: 'POST',
})
.then(response => {
if (response && response.success) {
this.$refs.editor.clear();
this.$toast({ type: 'success', message: '评论提交成功' });
this.$emit('append', response.data);
} else {
const errorMsg = response && response.message ? response.message : '评论提交失败';
this.$toast({ type: 'error', message: errorMsg });
}
})
.catch(error => {
const errorMsg = error && error.message ? error.message : '评论提交失败,请重试';
this.$toast({ type: 'error', message: errorMsg });
})
.finally(() => {
this.sending = false;
});
},
},
};