"images": "^3.0.2",
app.js
const koa = require('koa');
const Router = require('koa-router');
let koabody = require('koa-body');
const fs = require('fs')
const app = new koa();
const router = new Router();
const session = require('koa-session');
app.keys = ['some secret hurr'];
const path = require('path')
const static = require('koa-static');
//app.use(session(CONFIG, app));
const port = process.env.PORT || 5001;
router.get("/",async ctx => {
ctx.body = {msg:'Hello koa Interfaces'}
})
let Article = require('./model/Article');
let Comment = require('./model/Comment');
let User = require('./model/User');
router.get("/article/list",async ctx => {
const articles = await Article.findAll();
let rows = []
for(let i=0;i<articles.length;i++){
let row = articles[i].dataValues
console.log(row)
/*
let img = fs.readFileSync(row.id+'.base64','utf-8')
row.img = img
*/
rows.push(row)
}
console.log(rows)
ctx.body = rows
})
const mysqlApi = require('./model/mysqlApi')
const query = mysqlApi.query
router.post('/comment', async (ctx, next) => {
let row = ctx.request.body;
console.log(row)
let {articleid,content} = row
let sqlStr = `select * from comment where article_id = ${articleid} order by id asc LIMIT 1`
console.log(sqlStr)
let ret = await query( sqlStr )
console.log(ret)
console.log(ret.length)
let art ={}
let comment_no = 1
if(ret.length > 0){
comment_no = ret[0].comment_no+1
}
let tt = {
article_id:articleid,
content:content,
comment_no:comment_no
}
art = await Comment.create(tt)
ctx.body = art
})
router.get("/comment",async ctx => {
let request = ctx.request;
let req_query = request.query;
const id = req_query.id;
let sqlStr = `select * from comment where article_id = ${id} order by id asc `
console.log(sqlStr)
let ret = await query( sqlStr )
ctx.body = ret
})
router.get("/article",async ctx => {
let request = ctx.request;
let req_query = request.query;
const id = req_query.id;
const articles = await Article.findAll({where:{id:id}});
for(let i=0;i<articles.length;i++){
let row = articles[i].dataValues
console.log(row)
//let img = fs.readFileSync(row.id+'.base64','utf-8')
//row.img = img
//rows.push(row)
}
console.log(ctx.session.username)
console.log(id)
console.log(articles)
ctx.body = articles
})
let images = require("images");
router.post('/add', async (ctx, next) => {
let row = ctx.request.body;
console.log(row.title)
//console.log(row.img)
let tt = {
title:row.title,
content:row.content,
newsid:row.newsid
}
console.log(tt)
let art = await Article.create(tt)
let id = art.dataValues.id
let fileType = row.filetype;
//console.log(art.dataValues.id)
//fs.writeFileSync(art.dataValues.id+'.base64',row.img);
var base64Data = row.img.replace(/^, "");
let img = new Buffer(base64Data, 'base64');
fs.writeFileSync(`${id}.${fileType}`,img);
images("11.png")
//加载图像文件
.size(180)
.save(`${id}_small.jpg`, {
quality : 50 //保存图片到文件,图片质量为50
});
//if(row.name == 'admin' && row.password == 'admin')
ctx.response.body = {code:0};
//ctx.response.body = 'hah';
});
app.use(async (ctx, next) => {
console.log(`Process ${ctx.request.method} ${ctx.request.url}...1`);
console.log( ctx.session)
if(ctx.request.method=='post')
if( undefined == ctx.session.sesStr ){
ctx.response.body = {code:-1,sessionConnect:-1};
return;
}
await next();
});
router.post('/login', async (ctx, next) => {
let row = ctx.request.body;
console.log( ctx.request)
console.log( row)
console.log(row.name)
let name = row.name
//if(row.name == 'admin' && row.password == 'admin')
const user = await User.findAll({where:{name:name}});
if(user.length > 0)
{
console.log(user)
if(user[0].passwd == row.password){
ctx.session.username = name;
ctx.session.id = user.id;
ctx.response.body = {code:0};
}
else
ctx.response.body = {code:-1};
}else{
ctx.response.body = {code:-1};
}
//ctx.response.body = 'hah';
});
router.get("/test",async ctx => {
ctx.body ="1"
})
// 配置静态web服务的中间件
//app.use(static(path.join(__dirname,'./images')));
app.use(static(__dirname+'/images'));
app.use(koabody({
multipart: true,
formidable: {
maxFileSize: 200*1024*1024 // 设置上传文件大小最大限制,默认2M
},
formLimit:20*1024*1024,
}));
app.keys = ['some secret hurr'];
const CONFIG = {
key: 'koa:sess', //cookie key (default is koa:sess)
maxAge: 3600000, // cookie的过期时间 maxAge in ms (default is 1 days)
overwrite: true, //是否可以overwrite (默认default true)
httpOnly: true, //cookie是否只有服务器端可以访问 httpOnly or not (default true)
signed: true, //签名默认true
rolling: false, //在每次请求时强行设置cookie,这将重置cookie过期时间(默认:false)
renew: false, //(boolean) renew session when session is nearly expired,
};
app.use(session(CONFIG, app));
//app.use(static(__dirname+'/images'));
app.use(router.routes()).use(router.allowedMethods);
app.listen(port,() =>{
console.log(`server started on ${port}!`)
})
let Sequelize = require('sequelize');
let mySequelize = require('./configdb');
var Article = mySequelize.define('article', {
id:{ //自增长id,主键,整形
type:Sequelize.INTEGER,
primaryKey: true,
autoIncrement:true
},
title: { //名字
type: Sequelize.STRING(100)
},
content: { //名字
type: Sequelize.STRING(500)
},
image_url: { //名字
type: Sequelize.STRING(100)
},
source: { //名字
type: Sequelize.STRING(50)
},
newsid: { //类型 在tab区别
type: Sequelize.STRING(50)
},
comment_count: { //类型 在tab区别
type: Sequelize.INTEGER
},
},{
timestamps: true,
freezeTableName: true,
});
Article.sync(); //创建表
module.exports = Article
login
<template> <div > <van-nav-bar title="登陆" left-text="返回" left-arrow @click-left="onClickLeft" @click-right="onClickRight" /> <van-cell-group class="user-group" style="margin-top:50px;margin-bottom:10px"> <van-cell title="没有注册?先去注册" style="color: rgba(69, 90, 100, 0.6);" @click="onClickRegister" is-link /> </van-cell-group> <van-cell-group> <van-field v-model="username" required clearable label="用户名" right-icon="question-o" placeholder="请输入用户名" @click-right-icon="$toast('question')" /> <van-field v-model="password" type="password" label="密码" placeholder="请输入密码" required /> <van-cell > <van-button @click="onClick" type="primary" style="margin-top:30px;100%;" size="normal">登录</van-button> </van-cell> </van-cell-group> </div> </template> <script> import { api } from '../../common/api' export default { data() { return { username:'', password:'', ifWrite:false, }; }, mounted: function() { }, methods: { async onClick(){ //alert(this.username) const row = {name:this.username,password:this.password} let ret = await api.post('api/login?time='+new Date().getTime(),row,this) if(ret.code == 0){ this.$store.state.logined = true; this.$store.state.user = this.username // this.$router.push({path: '/list'}); } }, onClickLeft(){ this.$router.go(-1) }, onClickRegister(){ this.$router.push({path: 'register'}); } } }; </script>
user
<template> <div> <van-nav-bar title="用户" left-text="返回" left-arrow @click-left="onClickLeft" @click-right="onClickRight" /> <van-cell-group class="user-group" style=" margin-top:10%;padding:15px;padding-left:20px;"> <div style="display:flex;font-size: 12px;"> <div style="border:1px solid"> <svg class="icon" style=" 2em; height: 2em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="564"><path d="M982.862052 804.664259c-25.592502-60.582251-62.281753-114.966318-109.068046-161.752612S772.623645 559.536073 712.041394 533.843601c-23.693059-9.997071-47.885971-18.094699-72.478766-24.392854 28.891536-12.996193 55.383774-31.290833 78.477009-54.284097 50.085327-50.085327 77.577272-116.56585 77.577272-187.445084s-27.591916-137.359758-77.577272-187.445085C667.95431 30.291126 601.373816 2.699209 530.594552 2.699209S393.234794 30.291126 343.249439 80.376452 265.572196 196.942302 265.572196 267.721566s27.591916 137.359758 77.577272 187.445084c20.194084 20.194084 42.987406 36.689251 67.680172 49.185591-32.490481 6.698038-64.181197 16.495167-94.972176 29.59133-60.582251 25.592502-114.966318 62.281753-161.752612 109.068047S70.729279 744.082007 45.136776 804.664259C18.644538 867.445865 5.148492 934.026359 5.148492 1002.706238l0.09997 21.293762H1022.850337l0.099971-21.293762c-0.099971-68.679879-13.596017-135.260373-40.088256-198.041979zM305.56048 267.721566c0-124.063653 100.970419-225.034072 225.034072-225.034072s225.034072 100.970419 225.034072 225.034072-100.970419 225.034072-225.034072 225.034072-225.034072-100.970419-225.034072-225.034072zM45.536659 984.111686C55.333789 734.184907 261.673338 533.843601 513.999414 533.843601c252.326076 0 458.665625 200.241336 468.462755 450.168114H45.536659z" fill="#727171" p-id="565"></path></svg> </div> <span style="padding-left:20px;padding-top:7px"> admin </span> </div> </van-cell-group> <van-cell-group> <van-cell title="我发表的" value="10篇文章" is-link /> <van-cell title="我的手机号" value="15862365445" is-link /> </van-cell-group> <van-cell > <van-button @click="onClick" type="warning" style="margin-top:20px;100%;" size="normal">注销</van-button> </van-cell> <div style=" position:fixed;bottom:0;100%;"> <van-row class="user-links" > <van-col span="8"> 首页 </van-col> <van-col span="8"> 发文 </van-col> <van-col span="8"> 未登陆 </van-col> </van-row> </div> </div> </template> <script> import { Row, Col, Icon, Cell, CellGroup } from 'vant'; export default { components: { [Row.name]: Row, [Col.name]: Col, [Icon.name]: Icon, [Cell.name]: Cell, [CellGroup.name]: CellGroup }, methods: { onClickLeft(){ this.$router.go(-1) }, async onClick(){ this.$store.state.logined = false; this.$store.state.user = null this.$router.push({path: '/login'}); }, } }; </script>
add
<template> <div > <van-nav-bar title="写文章" left-text="返回" right-text="按钮" left-arrow @click-right="onClickRight" @click-left="onClickLeft" /> <van-cell-group> <van-field readonly clickable v-model=classValue placeholder="选择类别" @click="onClickField" /> <van-popup v-model="showPicker" position="bottom"> <van-picker show-toolbar :columns=columns @cancel="onCancel2" @confirm="onConfirm2" /> </van-popup> </van-cell-group> <van-cell-group> <van-field v-model=titleValue placeholder="请输入标题" bind:change="onChange" /> </van-cell-group> <van-cell-group> <van-cell-group> <div style="border:1px solid #ededed;"> <van-field style="height:250px" v-model=contentValue type="textarea" placeholder="请输入内容" autosize /> <van-uploader style="margin-top:5px;margin-left:5px" v-model="fileList" multiple :max-count="1" :after-read="afterRead" /> </div> </van-cell-group> </van-cell-group> <div style="position:fixed;bottom:0;100%;"> <van-cell-group> </van-cell-group> <van-row class="user-links" > <van-col span="8"> 首页 </van-col> <van-col span="8"> 发文 </van-col> <van-col span="8"> 未登陆 </van-col> </van-row> </div> </div> </template> <script> import { api } from '../../common/api' export default { data() { return { columns: ['杭州', '宁波', '温州', '嘉兴', '湖州'], showPicker: false, classValue: '', titleValue:'', contentValue:'', img:'', fileType:'', fileList: [] } }, methods: { oversize(){ alert('too big') }, async onClickRight(){ debugger; console.log(this.classValue,this.titleValue,this.contentValue) const row = {newsid:this.classValue, title:this.titleValue, content:this.contentValue, img:this.img, filetype:this.fileType} let ret = await api.post('api/add?time='+new Date().getTime(),row,this) if(ret.code == 0){ alert('上传成功') } }, onClickField() { this.showPicker = true; }, onConfirm2(value) { this.showPicker = false; this.classValue = value; }, onCancel2() { this.showPicker = false; }, afterRead(file) { // 此时可以自行将文件上传至服务器 console.log(file) let arr = file.file.name.split('.'); let fileType = arr[arr.length-1] console.log(fileType); //console.log(file.content); this.img = file.content this.fileType = fileType }, onClickLeft(){ this.$router.go(-1) } } } </script>
register
<van-nav-bar
title="注册"
left-text="返回"
left-arrow
@click-left="onClickLeft"
@click-right="onClickRight"
/>
<van-cell-group class="user-group" style="margin-top:20px;margin-bottom:10px">
<van-cell title="已注册?去登陆" style="color: rgba(69, 90, 100, 0.6);" is-link />
</van-cell-group>
detail
<template>
<div>
<van-nav-bar style=" position:fixed;top:0;100%;"
:title=options1.title
left-text="返回"
left-arrow
@click-left="onClickLeft"
/>
<van-cell-group style="margin-top:20px;margin-bottom:100px;">
<van-cell>
<h4>{{options1.title}} </h4>
<small>{{options1.author_name}} 发表于 {{options1.published_at}} </small>
</van-cell>
<van-cell>
<img style="100%;height:auto;" :src=options1.image_url >
{{options1.content}}
</van-cell>
<van-cell>
<small>评论{{options1.comments_count}}</small>
</van-cell>
<van-cell>
<div class="comment-item" v-for="item in comments">
<span class="first-line">
<span class="author">
{{item.comment_no}}# {{item.author}}
</span>
<span class="zan">{{item.zans}}赞</span>
</span>
<span class="content">
<div v-html="item.content"> </div>
</span>
<span class="info">
<span> {{item.time}}</span>
<span @click="onClickReply(item)">回复</span>
</span>
</div>
</van-cell>
</van-cell-group>
<van-cell class="foot">
<div v-show="!ifWrite" class="orgin-comment" @click="onClick()">
<span> 写评论...</span>
</div>
<div v-show="ifWrite" class="write-comment" >
<div style="border:1px solid #ededed;border-radius: 5px;100%">
<van-field
@blur='onBlur'
autofocus
ref='focusTextarea'
v-model="cmdtext"
type="textarea"
:placeholder=placeholder
autosize
/>
<svg class="icon" style="margin:5px; 2em; height: 2em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1196"><path d="M271.738 465.729c56.472 0 102.438-46.461 102.438-103.514 0-57.083-45.966-103.515-102.438-103.515S169.3 305.132 169.3 362.215C169.3 419.268 215.266 465.729 271.738 465.729zM271.738 288.5c40.058 0 72.637 33.074 72.637 73.715 0 40.654-32.579 73.714-72.637 73.714-40.059 0-72.638-33.06-72.638-73.714C199.1 321.574 231.679 288.5 271.738 288.5zM65 154.4 65 869.6l894 0L959 154.4 65 154.4zM929.2 839.8 123.582 839.8l288.76-244.322L560.57 724.965 929.2 430.355 929.2 839.8zM929.2 392.218 561.414 686.144 412.662 556.176 94.8 825.118 94.8 184.2l834.4 0L929.2 392.218z" p-id="1197"></path></svg>
</div>
<!--记录why不行??? -->
<span @click="onPost()" style="margin-left:5px;40px;line-height: 80px;">提交</span>
<span @click="onCancel()" style="margin-left:5px;40px;line-height: 80px;">取消</span>
</div>
</van-cell>
</div>
</template>
<script>
import { api } from '../../common/api'
export default {
data() {
return {
placeholder:"请输入评论",
options1: {
author_name: "36氪出海",
comments_count: 10,
content: "- Node.js是V..",
article_type: 1,
/* //cover:"https://pic.36krcnd.com/201910/25072501/lvaqh1a5mp8mduj3!heading",
image_url:
"https://pic.36krcnd.com/201910/25072501/lvaqh1a5mp8mduj3!heading",
id: 101711,
post_id: "5259650",
published_at: "2019-10-26 09:31:06",
title:
"出海创投周报 | 拉美独角兽 Rappi 进军哥斯达黎加;Gojek 表示将为双重上市做准备"
*/
},
comments: [{
comment_no:1,
zans:12,
author:'正阳山大猴子',
content:'如设置了width则元素尺寸由width决定',
time:'1天前',},
{ comment_no:2,
zans:2,
author:'正阳山大猴子',
content:' flex-basis为auto时'+' //<small><b> 1# @正阳山大猴子</b> </small>',
time:'1天前',
}],
ifWrite:false,
cmdtext:'',
replytxt:''
};
},
mounted: function() {
//this.options.image_url = this.options.cover
this.getData()
//alert('1,'+this.$route.query.id)
this.getComment()
},
methods: {
async onPost(){
//alert(this.cmdtext)
if(this.cmdtext=='') return;
let row = {articleid:this.$route.query.id,content:this.cmdtext+this.replytxt}
console.log(row)
let ret = await api.post('api/comment?time='+new Date().getTime(),row,this)
console.log('ret')
this.cmdtext = ''
this.ifWrite = false
},
async getComment(){
let ret = await api.get('api/comment?time='+new Date().getTime()+'&id='+this.$route.query.id)
console.log(ret)
this.comments = ret
},
async getData(){
let ret = await api.get('api/article?time='+new Date().getTime()+'&id='+this.$route.query.id)
let row = ret[0]
this.options1.image_url ='api/'+row.id+'.jpg'
this.options1.title = row.title
this.options1.content = row.content
console.log(ret)
},
onClick(){
this.ifWrite = true
this.$refs.focusTextarea.focus()
this.placeholder = '请输入内容'
this.replytxt = ''
console.log( this.$refs.focusTextarea.focused)
},
onClickReply(item){
this.ifWrite = true
this.placeholder = '回复'+item.author
this.replytxt = ` //<small><b> ${item.comment_no}# @${item.author}</b> </small>`
this.$refs.focusTextarea.focus()
//alert(item.no)
},
onFocus(){
//alert(1)
},
onBlur(){
//this.ifWrite = false
},
onCancel(){
this.ifWrite = false
},
onClickLeft(){
this.$router.go(-1)
}
}
};
</script>
<style lang="less">
.foot{
100%;
display: block;
border: 1px solid #ebebeb;
margin:0 auto;
//margin-bottom: 30px;
background:#fff;
position:fixed;
z-index: 99999;
bottom:0;
text-align:center;
.orgin-comment{
120px;
background-color:#eee ;
border-radius: 20px;
display: block;
font-size: 15px;
margin: 5px 5px;
}
.write-comment{
display: flex;
flex-direction: row;
}
}
.comment-line{
display: flex;
flex-direction: row;
}
.comment-item{
display: flex;
flex-direction: column;
margin-bottom:9px;
.first-line{
justify-content:space-between;
.comment-line();
.author{
color: rgb(30, 102, 204);
font-size: 13px;
}
.zan{
font-size: 11px;
color: gray;
}
}
.content{
padding-left: 5px;
font-size: 13px;
line-height: 18px;
small{
background-color: #eee;
}
}
.info{
padding-left: 5px;
.comment-line();
font-size: 10px;
color: gray;
span{
50%;
}
}
span{
padding-right:5px;
}
}
</style>