Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/rnd-soft/rails_spa


https://github.com/rnd-soft/rails_spa

Last synced: about 7 hours ago
JSON representation

Awesome Lists containing this project

README

        

# RailsSpa

Данный гем икапсулирует в себе следующие javascript-библиотеки и сторонние решения:

1. [AngualarJS v1.4.4](https://github.com/angular/angular.js/tree/v1.4.4) (ngRoute, ngResource, ngSanitize)
2. [UnderscoreJS v1.8.3](https://github.com/jashkenas/underscore/tree/1.8.3)
3. [dcbox](https://github.com/dcversus/dcbox)
4. [ng-notify](https://github.com/matowens/ng-notify)
4. [pace](https://github.com/HubSpot/pace)
5. [perfect-scrollbar](https://github.com/noraesae/perfect-scrollbar)

Гем не требует наличия jQuery в проекте. При этом всячески поощряется полный отказ от данной библиотеки. Перед использованием гема убедитесь, что в application.js у вас выключены turbolinks.

Для установки гема добавьте в Gemfile следующую строчку:

```
gem 'rails_spa', git: '[email protected]:storuky/rails_spa.git'
```
Следующим шагом требуется подключить компоненты в основной лейаут.
```
= render 'rails_spa/components'
```

Далее необходимо подключить js и css библиотеки в ваши assets-файлы(по умолчанию application.js и application.scss).

```
//= require rails_spa
```

Так же необходимо заинжектить AngularJS модуль в ваше приложение:
```
angular.module('app', ['rails_spa'])
```

Для корректной работы требует следующие гемы:

1. [js-routes](https://github.com/railsware/js-routes)
2. [slim-rails](https://github.com/slim-template/slim-rails)
3. [gon](https://github.com/gazay/gon)
4. [carrierwave](https://github.com/carrierwaveuploader/carrierwave)
5. [active_model_serializers](https://github.com/rails-api/active_model_serializers)

## 1. Структура приложения. Скаффолдинг
Для структурированности вашего приложения и его адекватной интеграции с Rails Assets Pipeline используйте следующую структуру:

**Пример 1.1** *Структура AngularJS приложения*
```
/javascripts
/controllers
/directives
/services
/filters
application.js
config.js
routes.js
```

Скаффолдинг позволяет значительно ускорить разработку, так как сразу создает все необходимые контроллеры, модели и сервисы необходимые для обеспечения функциональности CRUD.

**Пример 1.2** *Скаффолдинг*
```
rails g spa_scaffold post title:string description:text user_id:integer
```

## 2. Контроллеры
Хорошей практикой является наличие отдельного контроллера под каждую страницу. В случае похожести контроллеров, общие части следует выносить в сервисы.

**Пример 2.1** *Создание нового AngularJs-контроллера*
```
rails g ng_controller home
```

**Пример 2.2** *Пример сгенерированного AngularJs-контроллера*
```
app.controller('HomeCtrl', ['$scope', 'Page', function ($scope, Page) {
var ctrl = this;

Page.current = 'home';
}])
```

Сервис Page инкапсулирован внутри гема. Он отвечает за текущее состояние страницы. В нем удобно хранить название текущей страницы для подсветки соответствующего пункта в меню. Данный сервис находится в $rootScope и доступен внутри шаблнизатора как:

**Пример 2.3** *Сервис Page внутри шаблонизатора*
```
{{Page.current}}
```

Для навигации можно использовать следующий прием (синтаксис шаблонизатора [slim](https://github.com/slim-template/slim-rails))

**Пример 2.4** *Добавление класса active для текущей страницы*
```
nav
a href="/home" ng-class="{active: Page.current == 'home'}"
| Домашняя страница
```

## 3. Нотификация

Для того, чтобы отобразить пользователю сообщение, пришедшее с сервера, необходимо, чтобы в ответе присутствовало поле msg.

**Пример 3.1** *Рендеринг зеленого нотиса*
```
render json: {msg: "Все отлично"}
```

**Пример 3.2** *Рендеринг красного нотиса*
```
render json: {msg: "Запись не найдена"}, status: 404
```

## 4. Валидация форм

Для валидации форм необходимо, чтобы каждый элемент формы был обернут в тег с id, соответствующим названию данного поля в базе данных. Тогда в случае наличия в ответе поле errors будет произведена подсветка полей и под каждым полем будет выведена соответстующая ошибка.

**Пример 4.1** *Отображение ошибок на форме*
```
render json: {errors: @record.errors}, status: 422
```

**Пример 4.2** *Комбинация с нотификацией*
```
render json: {errors: @record.errors, msg: "Ошибка при сохранении"}, status: 422
```

## 5. Плюрализация

Написана фабрика и фильтр для плюрализации на русский язык.

**Пример 5.1** *Плюрализация внутри шаблнизатора*
```
{{comments.count | plur:["комментарий", "комментария", "комментариев"]}}
```
Так же может быть использован внутри AngularJs контроллеров, директив, сервисов и тд.

**Пример 5.2** *Плюрализация внутри котроллера/директивы/сервиса*
```
pluralize(count, ["комментарий", "комментария", "комментариев"])
```

## 6. Сервис авторизации
Работает в связке с devise. Позволяет обеспечить ajax-авторизацию и ajax-выход.

Для правильной работы сервиса необходимо сгенерировать devise-контроллеры командой:

**Пример 6.1** *Генерация devise-контроллеров для работы через ajax*
```
rails generate spa_devise users
```

В routes.rb необходимо указать эти контроллеры для devise_for

**Пример 6.2** *routes.rb для devise*
```
devise_for :users, controllers: {
sessions: 'users/sessions',
registrations: 'users/registrations'
}
```

Все готово для авторизации. Теперь создадим простую форму авторизации:

**Пример 6.3** *Форма авторизации*
```
form
input type="email" ng-model="Sign.user.email"
input type="password" ng-model="Sign.user.password"
button ng-click="Sign.in(Sign.user)"
```

**Пример 6.4** *Удаление сессии. Выход*

Все остальное сервис сделает за нас. Для удаления сессии необходимо вызвать метод Sign.out()
```
a ng-click="Sign.out()" Выход
```

## 7. Кастомный скролл
Для того, чтобы заменить браузерный скролл на кастомный, достатночно на нужном блоке объявить аттрибут scroll

**Пример 7.1** *Кастомный скролл*
```
div scroll="" rebuild="{{items}}" axis="y"
div ng-repeat="item in items"
```

**rebuild** – сообщает директиве о том, что скролл необходимо перестроить, так как размер контейнера изменился. axis указывает ось скрола.

## 8. dcbox
**dcbox** — отличная альтернатива для lightbox. Не требует никаких зависимостей, выглядит приятно, стабильно работает во всех современных браузерах.

**Пример 8.1** *dcbox*
```
.dcbox rebuild="{{images}}"
a href="{{image.url}}" ng-repeat="image in images"
img src="{{image.thumb.url}}"
```

**rebuild** – сообщает директиве о том, что dcbox необходимо перестроить.

## 9. Attachments
Строит список ссылок на файлы прикреплений в форме

**Пример 9.1** *Отображение прикреплений на форме*
```
attachments destroy-url="Routes.document_path" attachments="[{id: 1, href: '/file1', title: 'Файл 1'}, {id: 2, href: '/file2', title: 'Файл 2'}]"
```
Для задания такого формата используйте gem [active_model_serializers](https://github.com/rails-api/active_model_serializers)

## 10. Слайдер изображений

**Пример 10.1** *Пример слайдера с шириной 4 слайда*
```
slider max-slides="4" destroy-url="Routes.image_path" slides="[{id: 1, original: '/image1.png', thumb: '/image1-thumb.png'}, {id: 2, original: '/image2.png', thumb: '/image2-thumb.png'}]"
```
Для задания такого формата используйте gem [active_model_serializers](https://github.com/rails-api/active_model_serializers)

## 11. TinyMCE
Требует подключения скриптов и стилей TinyMCE 4

**Пример 11.1** *AngularJS TinyMCE 4*
```
textarea.tinymce ng-model="content"
```

## 12. Загрузка файлов на сервер

**Пример 12.1** *Реализация ng-model для input type="file" с автозагрузкой на сервер*
```
input type="file" multiple="true" fileupload="Routes.images_path()" ng-model="ctrl.initiative.documents"
```
ng-model сочетается с директивами attachments и slider.

Хорошей практикой является предзагрузка файлов на сервер до отправки самой формы. Сгенерирум аплоадер, сериалайзер, модель, миграцию и контроллер для загрузки файла следующей командой

**Пример 12.2**

```
rails g spa_uploader image
```

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

**Пример 12.3** *Пример контроллера, создающего связь для загруженных файлов*
```
class PostsController < ApplicationController
def create
@post = current_user.initiatives.new post_params

if @post.save
associate :images, for: @post
render json: {msg: "Пост успешно создан!"}
else
render json: {errors: @post.errors}, status: 422
end
end

end
```

Метод associate инкапсулирован внутри гема и доступен внутри любого контроллера, унаследованного от ActionController::Base. Принимает так же поля в качестве массива

**Пример 12.4** *Передача массива прикреплений*

```
associate [:images, :documents], for: @post
```

## 13. Принцип построения SPA-приложения
Построение рельсового SPA-приложение начинается с редактирования routes.rb

**Пример 13.1** *routes.rb для spa-приложения*

```
devise_for :users, controllers: {
sessions: 'users/sessions',
registrations: 'users/registrations'
}

root 'application#index'

scope :ajax do
get 'home' => 'Home#index'
resource :posts
# Здесь весь роутинг текущего приложения
end

get '/*path' => 'application#index'
```

Мы помещаем все роуты внутрь scope :ajax, тем самым добавляя к прослушиваемым роутам префикс /ajax. Это необходимо для того, чтобы при нажатии кнопки F5 на странице /posts нам не возвращался template без лейаута, а возвращалась индексная страница приложения к которой AngularJS применит свой html5-роутинг.

Индексная страница приложения находится в ApplicationController. При этом здесь же мы выключаем рельсовый layout

**Пример 13.2** *application_controller.rb для spa-приложения*

```
class ApplicationController < ActionController::Base
protect_from_forgery with: :exception

layout :false

def index
render template: "layouts/application"
end
end
```

Далее необходимо внутри вьхи views/layouts/application.html.slim зменить =yield на ng-view. Так же необходимо прописать внутри тега head тег

**Пример 13.3** *Тег base для spa-приложения*
```

```
Все готово для того, чтобы строить spa-роутинг приложения. Создать его очень просто. Достаточно воспользоваться генератором

**Пример 13.4** *Генератор роутинга*
```
rails g ng_routes
```