在 AJAX 调用(使用 ExpressJS)之后循环遍历 EJS 模板中的数组的正确方法是什么?
What is the proper way to loop through an array in an EJS template after an AJAX Call (using ExpressJS)?
因此,我尝试使用 request
module/package 遍历我从使用内部 API 的 http 调用获得的对象数组。到目前为止,我能够从 API 取回我的数据并在我的页面上显示完整的对象。我想在我的页面上显示它并使用 EJS 模板系统循环它。我知道我可以将 AngularJS 用于前端,但我想看看仅使用服务器端我能走多远。
下面是我的代码:
server.js
// Prepend /api to my apiRoutes
app.use('/api', require('./app/api'));
api.js
var Report = require('./models/report');
var express = require('express');
var apiRoutes = express.Router();
apiRoutes.route('/reports', isLoggedIn)
.get(function (req, res,next) {
// use mongoose to get all reports in the database
Report.find(function (err, reports) {
// if there is an error retrieving, send the error.
// nothing after res.send(err) will execute
if (err)
return res.send(err);
res.json(reports);
});
});
routes.js
var User = require('./models/user');
var request = require('request');
module.exports = function (app, passport) {
app.get('/reports', isLoggedIn, function (req, res) {
res.render('pages/new-report.ejs', {
user: req.user,
title:'New Report'
});
});
request({
uri:'http://localhost:2016/api/reports',
method:'GET'
}).on('data',function(data){
console.log('decoded chunk:' + data)
}).on('response',function(resp){
resp.on('data', function(data){
console.log('received:' + data.length + ' bytes of compressed data');
app.get('/timeline', isLoggedIn, function (req, res) {
res.render('pages/timeline', {
user: req.user,
title:'Timeline',
reports: data
});
});
})
});
}
reports.ejs
因此,如果我简单地在我的页面上像这样 <p><%= reports %></p>
输出整个 reports
对象,一切正常,我得到如下内容:
[
{
"_id": "5775d396077082280df0fbb1",
"author": "57582911a2761f9c77f15528",
"dateTime": "30 June 2016 - 07:18 PM",
"picture": "",
"description": "",
"location": [
-122.46596999999997,
37.784495
],
"latitude": 37.784495,
"longitude": -122.46596999999997,
"address": "4529 California St, San Francisco, CA 94118, USA",
"contact": "John Doe",
"type": "Financial",
"__v": 0,
"updated_at": "2016-07-01T02:21:10.259Z",
"created_at": "2016-07-01T02:21:10.237Z",
"tags": [
"tag1,tag2"
]
}
]
但是,如果我尝试如下所示遍历对象,它会得到一个 UNDEFINED 作为我的 的 return 属性 report 对象,我显然得到了一个无限循环。
<ul class="timeline">
<% reports.forEach(function(report) { %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= report.type %> </span>
<span class="time" style="font-weight:700;font-size:25px;line-height:20px;text-align:left;">
<%= report.dateTime %> </span>
</div>
</li>
<% }) %>
</ul>
我已经尝试了循环的另一种变体,但我仍然不成功:
<ul class="timeline">
<% for (var i = 0; i < reports.length; i++) { %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= report[i].type %>
4/10/13 </span>
<span class="time" style="font-weight:700;font-size:25px;line-height:20px;text-align:left;">
<%= report[i].dateTime %> </span>
</div>
</li>
<% } %>
</ul>
ejs
中 for
循环的语法是完美的,但是迭代数组名称是 reports 而您似乎使用 迭代中的 report[i],需要将其更改为 reports[i],这应该有效。
reports.ejs
<ul class="timeline">
<% for (var i = 0; i < reports.length; i++) { %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= reports[i].type %>
4/10/13 </span>
<span class="time" style="font-weight:700;font-size:25px;line-height:20px;text-align:left;">
<%= reports[i].dateTime %> </span>
</div>
</li>
<% } %>
</ul>
希望对您有所帮助。
I guess something like this ..
<% if (reports.length > 0){%> // Checking if there are reports
<ul class="timeline">
<% for (let report of reports){ %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= report.type %>
4/10/13 </span>
<span class="time" style="font-weight:700;font-size:25px;line-
height:20px;text-align:left;">
<%= report.dateTime %> </span>
</div>
</li>
<% } %>
</ul>
<%}%>
<%}%>
这是我使用 loopback3 和 ejs 的工作版本:
在server/boot/routes.js:
module.exports = function(app) {
const router = app.loopback.Router();
router.get('/', function(req, res){
app.models.ZenGarden.find()
.then(plants => {
console.log('plants: ', plants)
res.render('index', {plants:plants})
}).catch(err => {
console.log('Failed to find in ZenGarden: ', err)
res.render('index')
})
});
router.post('/', function(req, res){
var plants = req.body.plants;
if (plants) {
for (var i = 0; i < plants.length; i++) {
console.log(plants[i])
app.models.ZenGarden.upsert(plants[i])
.then().catch(err => console.log(err))
}
}
return res.render('index', {plants:plants})
})
app.use(router);
};
在server/views/index.ejs:
<div>
<form action='/' method='POST'>
<% if (plants) { %>
<table>
<tr><td>Name</td><td>Count</td></tr>
<% for (var i = 0; i < plants.length; i++) { %>
<tr>
<input type='hidden' value=<%= plants[i].id %> name='plants[<%= i %>][id]'>
<input type='hidden' value="<%= plants[i].name %>" name='plants[<%= i %>][name]'>
<td><%= plants[i].name %></td>
<td><input type='text' value=<%= plants[i].count %> name='plants[<%= i%>][count]'</td></tr>
<% } %>
</table>
<button type='submit'>Save</button>
<% } else { %>
<p>No plant in Zen Garden :-(</p>
<% } %>
</form>
</div>
确保在 server/server.js 中添加以下内容:
const path = require('path')
const bodyparser = require('body-parser')
app.set('view engine', 'ejs')
app.set('views', path.resolve(__dirname, 'views'))
app.middleware('initial', bodyparser.urlencoded({extended:true}))
app.middleware('initial', bodyparser.json())
这里是模型ZenGarden(common/models/zen-garden.json)的定义:
{
"name": "ZenGarden",
"base": "PersistedModel",
"idInjection": true,
"options": {
"validateUpsert": true
},
"properties": {
"name": {
"type": "string",
"required": true
},
"count": {
"type": "number",
"required": true
}
},
"validations": [],
"relations": {},
"acls": [],
"methods": {}
}
因此,我尝试使用 request
module/package 遍历我从使用内部 API 的 http 调用获得的对象数组。到目前为止,我能够从 API 取回我的数据并在我的页面上显示完整的对象。我想在我的页面上显示它并使用 EJS 模板系统循环它。我知道我可以将 AngularJS 用于前端,但我想看看仅使用服务器端我能走多远。
下面是我的代码:
server.js
// Prepend /api to my apiRoutes
app.use('/api', require('./app/api'));
api.js
var Report = require('./models/report');
var express = require('express');
var apiRoutes = express.Router();
apiRoutes.route('/reports', isLoggedIn)
.get(function (req, res,next) {
// use mongoose to get all reports in the database
Report.find(function (err, reports) {
// if there is an error retrieving, send the error.
// nothing after res.send(err) will execute
if (err)
return res.send(err);
res.json(reports);
});
});
routes.js
var User = require('./models/user');
var request = require('request');
module.exports = function (app, passport) {
app.get('/reports', isLoggedIn, function (req, res) {
res.render('pages/new-report.ejs', {
user: req.user,
title:'New Report'
});
});
request({
uri:'http://localhost:2016/api/reports',
method:'GET'
}).on('data',function(data){
console.log('decoded chunk:' + data)
}).on('response',function(resp){
resp.on('data', function(data){
console.log('received:' + data.length + ' bytes of compressed data');
app.get('/timeline', isLoggedIn, function (req, res) {
res.render('pages/timeline', {
user: req.user,
title:'Timeline',
reports: data
});
});
})
});
}
reports.ejs
因此,如果我简单地在我的页面上像这样 <p><%= reports %></p>
输出整个 reports
对象,一切正常,我得到如下内容:
[
{
"_id": "5775d396077082280df0fbb1",
"author": "57582911a2761f9c77f15528",
"dateTime": "30 June 2016 - 07:18 PM",
"picture": "",
"description": "",
"location": [
-122.46596999999997,
37.784495
],
"latitude": 37.784495,
"longitude": -122.46596999999997,
"address": "4529 California St, San Francisco, CA 94118, USA",
"contact": "John Doe",
"type": "Financial",
"__v": 0,
"updated_at": "2016-07-01T02:21:10.259Z",
"created_at": "2016-07-01T02:21:10.237Z",
"tags": [
"tag1,tag2"
]
}
]
但是,如果我尝试如下所示遍历对象,它会得到一个 UNDEFINED 作为我的 的 return 属性 report 对象,我显然得到了一个无限循环。
<ul class="timeline">
<% reports.forEach(function(report) { %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= report.type %> </span>
<span class="time" style="font-weight:700;font-size:25px;line-height:20px;text-align:left;">
<%= report.dateTime %> </span>
</div>
</li>
<% }) %>
</ul>
我已经尝试了循环的另一种变体,但我仍然不成功:
<ul class="timeline">
<% for (var i = 0; i < reports.length; i++) { %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= report[i].type %>
4/10/13 </span>
<span class="time" style="font-weight:700;font-size:25px;line-height:20px;text-align:left;">
<%= report[i].dateTime %> </span>
</div>
</li>
<% } %>
</ul>
ejs
中 for
循环的语法是完美的,但是迭代数组名称是 reports 而您似乎使用 迭代中的 report[i],需要将其更改为 reports[i],这应该有效。
reports.ejs
<ul class="timeline">
<% for (var i = 0; i < reports.length; i++) { %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= reports[i].type %>
4/10/13 </span>
<span class="time" style="font-weight:700;font-size:25px;line-height:20px;text-align:left;">
<%= reports[i].dateTime %> </span>
</div>
</li>
<% } %>
</ul>
希望对您有所帮助。
I guess something like this ..
<% if (reports.length > 0){%> // Checking if there are reports
<ul class="timeline">
<% for (let report of reports){ %>
<li class="timeline-yellow">
<div class="timeline-time">
<span class="date" style="text-align:left">
<%= report.type %>
4/10/13 </span>
<span class="time" style="font-weight:700;font-size:25px;line-
height:20px;text-align:left;">
<%= report.dateTime %> </span>
</div>
</li>
<% } %>
</ul>
<%}%>
<%}%>
这是我使用 loopback3 和 ejs 的工作版本:
在server/boot/routes.js:
module.exports = function(app) {
const router = app.loopback.Router();
router.get('/', function(req, res){
app.models.ZenGarden.find()
.then(plants => {
console.log('plants: ', plants)
res.render('index', {plants:plants})
}).catch(err => {
console.log('Failed to find in ZenGarden: ', err)
res.render('index')
})
});
router.post('/', function(req, res){
var plants = req.body.plants;
if (plants) {
for (var i = 0; i < plants.length; i++) {
console.log(plants[i])
app.models.ZenGarden.upsert(plants[i])
.then().catch(err => console.log(err))
}
}
return res.render('index', {plants:plants})
})
app.use(router);
};
在server/views/index.ejs:
<div>
<form action='/' method='POST'>
<% if (plants) { %>
<table>
<tr><td>Name</td><td>Count</td></tr>
<% for (var i = 0; i < plants.length; i++) { %>
<tr>
<input type='hidden' value=<%= plants[i].id %> name='plants[<%= i %>][id]'>
<input type='hidden' value="<%= plants[i].name %>" name='plants[<%= i %>][name]'>
<td><%= plants[i].name %></td>
<td><input type='text' value=<%= plants[i].count %> name='plants[<%= i%>][count]'</td></tr>
<% } %>
</table>
<button type='submit'>Save</button>
<% } else { %>
<p>No plant in Zen Garden :-(</p>
<% } %>
</form>
</div>
确保在 server/server.js 中添加以下内容:
const path = require('path')
const bodyparser = require('body-parser')
app.set('view engine', 'ejs')
app.set('views', path.resolve(__dirname, 'views'))
app.middleware('initial', bodyparser.urlencoded({extended:true}))
app.middleware('initial', bodyparser.json())
这里是模型ZenGarden(common/models/zen-garden.json)的定义:
{
"name": "ZenGarden",
"base": "PersistedModel",
"idInjection": true,
"options": {
"validateUpsert": true
},
"properties": {
"name": {
"type": "string",
"required": true
},
"count": {
"type": "number",
"required": true
}
},
"validations": [],
"relations": {},
"acls": [],
"methods": {}
}