AngularJS - шаблоны не работают

Я использую Restangular для создания простого API с использованием стека MEAN.

Вот мой код:

index.html

<!DOCTYPE html>

<html data-ng-app="scotchTodo">
<head>
    <!-- META -->
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1"><!-- Optimize mobile viewport -->

    <title>Node/Angular Todo App</title>

    <!-- SCROLLS -->
    <link rel="stylesheet" href="//netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap.min.css"><!-- load bootstrap -->
    <style>
        html{
            overflow-y:scroll;
        }
        body{
            padding-top:50px; 
        }
    </style>    

    <!-- SPELLS -->
    <script src="//ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>

    <script src="//ajax.googleapis.com/ajax/libs/angularjs/1.2.15/angular.min.js"></script>
    <script src="//ajax.googleapis.com/ajax/libs/angularjs/1.2.15/angular-route.min.js"></script>

    <script src="//cdnjs.cloudflare.com/ajax/libs/restangular/1.4.0/restangular.min.js"></script>
    <script src="//cdnjs.cloudflare.com/ajax/libs/underscore.js/1.6.0/underscore-min.js"></script>

    <script src="app.js"></script>
</head>
<body>
    <div data-ng-view>

    </div>
</body>
</html>

app.js

var scotchTodo = angular.module('scotchTodo', ['restangular','ngRoute']);

//config
scotchTodo.config(['$routeProvider','$locationProvider', function($routeProvider, $locationProvider) {

    $routeProvider
    .when('/',{
        templateUrl: 'list.html',
        controller: 'ListController'
    })
    .when('api/todos/:todo_id',{
        templateUrl: 'edit.html',
        controller: 'EditController'
    });

    $locationProvider.html5Mode(true);

}]);


//controllers
scotchTodo.controller('ListController', ['$scope', 'Restangular',
    function($scope, Restangular) {

        //GET ALL
        var baseTodo = Restangular.all('api/todos');
        baseTodo.getList().then(function(todos) {
            $scope.todos = todos;
        });

        //POST -> Save new
        $scope.save = function() {
            var baseTodo = Restangular.all('api/todos');
            var newTodo = {'text': $scope.text};
            baseTodo.post(newTodo).then(function(todos) {
                $scope.todos = todos;
                $scope.text = '';
            });
        };

        //DELETE
        $scope.delete = function(id) {
            var baseTodo = Restangular.one('api/todos', id);
            baseTodo.remove().then(function(todos) {
                $scope.todos = todos;
            });
        };

    }]);

scotchTodo.controller('EditController', ['$scope', 'Restangular','$routeParams',
    function($scope, Restangular, $routeParams) {   

        var baseTodo = Restangular.one('api/todos', id);

        baseTodo.getList().then(function(todo) {
            $scope.todo = todo[0];
            window.test = "dev";
        });

        //PUT -> Edit
        $scope.update = function(id){
            var baseTodo = Restangular.one('api/todos', id);
            baseTodo.text = "Edited";
            baseTodo.put().then(function(todos) {
                $scope.todos = todos;
            });
        };

    }]);

list.html

<div>
    <div data-ng-repeat="todo in todos">
        {{todo.text}}<a href="api/todos/{{todo._id}}">Edit</a><button data-ng-click="delete(todo._id)">X</button>
    </div>
    <input type="text" data-ng-model="text"/>
    <button data-ng-click="save()">Add</button>
</div>

edit.html

<div>
    <input type="text" data-ng-model="text" value="{{todo.text}}" />
    <button data-ng-click="update(todo._id)">Save</button>
</div>

server.js

// setup ========================
var express = require('express');
var app = express();
var mongoose = require('mongoose');
var bodyParser = require('body-parser');

//configuration =================
mongoose.connect('mongodb://127.0.0.1:27017/sl', function(err, db) {
    if (!err) {
        console.log("We are connected to " + db);
    }
});

app.use(express.static(__dirname + '/public'));
app.use(bodyParser());

// application -------------------------------------------------------------
app.get('/', function(req, res) {
    res.sendfile('./public/index.html'); // load the single view file (angular will handle the page changes on the front-end)
});

//listen ========================
app.listen(8080);
console.log('App started on the port 8080');

//define model ==================
var Todo = mongoose.model('Todo', {
    text: String
});

// routes ======================================================================

// api ---------------------------------------------------------------------

//get one todo
app.get('/api/todos/:todo_id', function(req, res) {

    // use mongoose to get all todos in the database
    Todo.find({
        _id: req.params.todo_id
    },function(err, todos) {

        // if there is an error retrieving, send the error. nothing after res.send(err) will execute
        if (err){
            res.send(err);
        }

        res.json(todos); // return all todos in JSON format
    });
});

// get all todos
app.get('/api/todos', function(req, res) {

    // use mongoose to get all todos in the database
    Todo.find(function(err, todos) {

        // if there is an error retrieving, send the error. nothing after res.send(err) will execute
        if (err){
            res.send(err);
        }

        res.json(todos); // return all todos in JSON format
    });
});

// create todo and send back all todos after creation
app.post('/api/todos', function(req, res) {

    // create a todo, information comes from AJAX request from Angular
    Todo.create({
        text: req.body.text,
        done: false
    }, function(err, todo) {
        if (err){
            res.send(err);
        }

        // get and return all the todos after you create another
        Todo.find(function(err, todos) {
            if (err)
                res.send(err);
            res.json(todos);
        });
    });

});

// update todo and send back all todos after creation
app.put('/api/todos/:todo_id', function(req, res) {

    // create a todo, information comes from AJAX request from Angular
    Todo.update({
        _id: req.params.todo_id
    }, {
        text:req.body.text
    }, function(err, todo) {
        if (err){
            res.send(err);
        }

        // get and return all the todos after you create another
        Todo.find(function(err, todos) {
            if (err)
                res.send(err);
            res.json(todos);
        });
    });

});


// delete a todo
app.delete('/api/todos/:todo_id', function(req, res) {
    Todo.remove({
        _id: req.params.todo_id
    }, function(err, todo) {
        if (err){
            res.send(err);
        }

        // get and return all the todos after you create another
        Todo.find(function(err, todos) {
            if (err){
                res.send(err);
            }
            res.json(todos);
        });
    });
});

Первая страница моего приложения загружается отлично. Вот скриншот.

list.html

Но когда я нажимаю на любую из ссылок редактирования, она должна загрузить шаблон edit.html. Но он показывает пустую страницу без ошибок в консоли. Вот скриншот.edit.html

Я не могу понять, что не так. Пожалуйста помоги. Пожалуйста, спросите, нужен ли какой-либо другой фрагмент кода. Я добавил почти все, что я сделал. Я знаю, что это раздражает и не рекомендуется, но я не уверен, какая часть моего кода вызывает эту проблему.

РЕДАКТИРОВАТЬ 1:

Я предпочитаю, что URL для edit.html может быть не разрешен правильно. Но я не уверен, как это проверить! Любая помощь будет оценена.

РЕДАКТИРОВАТЬ 2: Структура каталогов

Структура каталогов

РЕШЕНИЕ: Вежливость @ashu

Проблема была в этой строке в index.html

<script src="app.js"></script>

Так должно быть:

<script src="/app.js"></script>

Однако мне не понятно почему! Страница в любом случае включала app.js. Это странно.

1 ответ

Решение

У вас есть одинаковые маршруты для угловых и экспресс.

.when('api/todos/:todo_id',{
    templateUrl: 'edit.html',
    controller: 'EditController'
});

и в экспресс

app.get('/api/todos/:todo_id', function(req, res) {

Отсюда и двусмысленность. Вы можете удалить часть "api" из угловых ссылок.

.when('/todos/:todo_id', {
     templateUrl: 'edit.html',
     controller: 'EditController'
 })

А на сервере вы можете добавить универсальный маршрут, который будет обрабатывать все не-API URL-адреса. Для этого вы можете переместить app.get('/', function(req,res) {..}) позвоните внизу после определения ваших маршрутов API.

// < Define APi Routes here >

//catch all route for serving the html template
app.get('/*', function(req, res ) {
      res.sendfile('./public/index.html')
});

Также, в вашем app.js EditController вы забыли инициализировать значение id.

var id = $routeParams.todo_id;
Другие вопросы по тегам