├── .gitignore
├── src
├── img
│ └── battlefield-js.jpg
├── js
│ ├── metrika.js
│ ├── battlefield.options.js
│ ├── battlefield.class-field.js
│ ├── battlefield.helper.js
│ ├── battlefield.class-battlefield.js
│ ├── battlefield.class-battle.js
│ └── battlefield.class-gameui.js
└── css
│ ├── battlefield.modal-window.css
│ ├── battlefield.player-name.css
│ ├── battlefield.logger.css
│ ├── battlefield.global.css
│ ├── battlefield.fields.css
│ └── battlefield.helper.css
├── bower.json
├── test
├── index.html
└── spec
│ └── class.battlefield.js
├── package.json
├── README.md
├── examples
├── index.html
└── development.html
├── gulpfile.js
└── dist
├── battlefield.min.js
└── battlefield.min.css
/.gitignore:
--------------------------------------------------------------------------------
1 | .idea
2 | bower_components
3 | node_modules
4 | npm-debug.log
--------------------------------------------------------------------------------
/src/img/battlefield-js.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/alx2das/battlefield-js/HEAD/src/img/battlefield-js.jpg
--------------------------------------------------------------------------------
/bower.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "battlefield-js",
3 | "description": "This is classic game Battlefield for browsers, implemented on language JavaScript.",
4 | "main": "examples/index.html",
5 | "authors": [
6 | "Alexandr Builov"
7 | ],
8 | "license": "ISC",
9 | "keywords": [
10 | "battlefield",
11 | "javascript",
12 | "gulp",
13 | "jasmine"
14 | ],
15 | "homepage": "https://github.com/alx2das/battlefield-js",
16 | "ignore": [
17 | "**/.*",
18 | "node_modules",
19 | "bower_components",
20 | "test",
21 | "tests"
22 | ],
23 | "dependencies": {
24 | "jasmine-core": "jasmine#^2.5.2"
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/src/js/metrika.js:
--------------------------------------------------------------------------------
1 | (function (d, w, c) {
2 | (w[c] = w[c] || []).push(function () {
3 | try {
4 | w.yaCounter42568409 = new Ya.Metrika({
5 | id: 42568409,
6 | clickmap: true,
7 | trackLinks: true,
8 | accurateTrackBounce: true
9 | });
10 | } catch (e) {
11 | }
12 | });
13 |
14 | var n = d.getElementsByTagName("script")[0],
15 | s = d.createElement("script"),
16 | f = function () {
17 | n.parentNode.insertBefore(s, n);
18 | };
19 | s.type = "text/javascript";
20 | s.async = true;
21 | s.src = "https://mc.yandex.ru/metrika/watch.js";
22 |
23 | if (w.opera == "[object Opera]") {
24 | d.addEventListener("DOMContentLoaded", f, false);
25 | } else {
26 | f();
27 | }
28 | })(document, window, "yandex_metrika_callbacks");
--------------------------------------------------------------------------------
/test/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Battlefield JS
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "battlefield-js",
3 | "version": "1.0.1",
4 | "description": "This is classic game Battlefield for browsers, implemented on language JavaScript.",
5 | "main": "examples/index.html",
6 | "directories": {
7 | "example": "examples",
8 | "test": "test"
9 | },
10 | "dependencies": {},
11 | "devDependencies": {
12 | "autoprefixer": "^6.7.2",
13 | "bower": "^1.8.0",
14 | "del": "^2.2.2",
15 | "gulp": "^3.9.1",
16 | "gulp-concat": "^2.6.1",
17 | "gulp-minify-css": "^1.2.4",
18 | "gulp-postcss": "^6.3.0",
19 | "gulp-rename": "^1.2.2",
20 | "gulp-uglify": "^2.0.0",
21 | "run-sequence": "^1.2.2"
22 | },
23 | "scripts": {
24 | "test": "test/index.html"
25 | },
26 | "repository": {
27 | "type": "git",
28 | "url": "git+https://github.com/alx2das/battlefield-js.git"
29 | },
30 | "keywords": [
31 | "battlefield",
32 | "javascript",
33 | "gulp",
34 | "jasmine"
35 | ],
36 | "author": "Alexandr Builov",
37 | "license": "ISC",
38 | "bugs": {
39 | "url": "https://github.com/alx2das/battlefield-js/issues"
40 | },
41 | "homepage": "https://alx2das.github.io/battlefield-js/examples/"
42 | }
43 |
--------------------------------------------------------------------------------
/src/css/battlefield.modal-window.css:
--------------------------------------------------------------------------------
1 | .bf-modal-font {
2 | position: fixed;
3 | z-index: 9;
4 | top: 0;
5 | left: 0;
6 | width: 100%;
7 | height: 100%;
8 | background: rgba(122, 139, 159, 0.8);
9 | }
10 |
11 | .bf-modal-window {
12 | position: fixed;
13 | z-index: 19;
14 | top: 50%;
15 | left: 50%;
16 | width: 600px;
17 | background: #fff;
18 | border-radius: 4px;
19 | }
20 |
21 | .bf-modal-window .content {
22 | padding: 25px 15px 15px 15px;
23 | }
24 |
25 | .bf-modal-window .content span {
26 | float: right;
27 | font-size: 12px;
28 | font-weight: 400;
29 | padding: 2px 0;
30 | color: #7A8BB4;
31 | font-style: italic;
32 | }
33 |
34 | .bf-modal-window .header, .bf-modal-window .cont, .bf-modal-window .footer {
35 | }
36 |
37 | .bf-modal-window .header {
38 | font-weight: 700;
39 | font-size: 16px;
40 | padding-bottom: 16px;
41 | border-bottom: 2px solid rgba(122, 139, 159, .3);
42 | }
43 |
44 | .bf-modal-window .cont {
45 | padding: 30px 0;
46 | }
47 |
48 | .bf-modal-window .footer {
49 | padding-top: 15px;
50 | border-top: 2px solid rgba(122, 139, 159, .3);
51 | }
52 |
53 | .bf-modal-window .header:after, .bf-modal-window .cont:after, .bf-modal-window .footer:after {
54 | content: " ";
55 | clear: both;
56 | display: block;
57 | }
--------------------------------------------------------------------------------
/src/css/battlefield.player-name.css:
--------------------------------------------------------------------------------
1 | .bf-player-name {
2 | text-align: center;
3 | width: 950px;
4 | margin: 0 auto
5 | }
6 |
7 | .bf-player-name span {
8 | display: inline-block;
9 | padding: 0 10px;
10 | font-size: 15px;
11 | height: 24px;
12 | line-height: 24px;
13 | }
14 |
15 | .bf-player-name .name {
16 | width: 120px;
17 | background: #7a8b9f;
18 | color: #fff;
19 | margin: 0 10px 0 6px;
20 | padding: 4px 10px;
21 | border-radius: 3px;
22 | height: auto;
23 | line-height: 1.1;
24 | }
25 |
26 | .bf-player-name .name.act {
27 | background: #f0ad4e;
28 | }
29 |
30 | .bf-player-name .total {
31 | min-width: 30px;
32 | text-align: center;
33 | color: #7a8b9f;
34 | font-weight: 700;
35 | }
36 |
37 | .bf-player-name .opt {
38 | float: left
39 | }
40 |
41 | .bf-player-name .left {
42 | float: left;
43 | margin-top: -5px
44 | }
45 |
46 | .bf-player-name .center {
47 | width: 460px;
48 | margin: 0 auto
49 | }
50 |
51 | .bf-player-name .center:after {
52 | content: " ";
53 | clear: both;
54 | display: block;
55 | }
56 |
57 | .bf-player-name .right {
58 | float: right
59 | }
60 |
61 | .bf-player-name .left a {
62 | color: #7a8b9f;
63 | font-weight: 700;
64 | font-size: 26px;
65 | text-decoration: none;
66 | }
67 |
68 | .bf-player-name .right span {
69 | padding: 0
70 | }
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Battlefield 1.0.1
2 | Игра была разработанна шутки ради, практики для, в свободное от основной работы время и помере желания выседания перед компьютером. Алгоритмы игры стараются максимально соответствовать правилам классической игры "Морской бой". Присутствует подобие искуственного интелекта (не обучается), старается сделать минимальное количество выстрелов для победы.
3 |
4 | > Игра может не запуститься в браузерах Internet Explorer, при чем даже в 11 верии,
5 | > все за за `Object.assign()` и многих других мелочей которые не поддерживаются в IE.
6 |
7 |
8 | ## Примеры
9 | [Основной пример можно мосмотреть GitHub Pages](https://alx2das.github.io/battlefield-js/examples/).
10 |
11 |
12 | ## Сборка
13 | Для сборки у вас уже должен быть установлен [Node.js с NPM](https://nodejs.org/). Все необходимые зависимости приписаны в файле `package.json`.
14 | Я использовал [Gulp](http://gulpjs.com/), поэтому у Вас он должен быть установлен глобально.
15 |
16 | ```sh
17 | npm i gulp -g
18 | ```
19 |
20 | Для запуска используйте команду по умолчанию:
21 |
22 | ```sh
23 | gulp
24 | ```
25 |
26 |
27 | ## Тестирование
28 | Для тестирования я использовал [Jasmine](https://jasmine.github.io/2.0/introduction.html). Пока что реализованно только браузерное тестирование, пожже добавлю серверное при сборке.
29 |
30 | Для запуска тестов должен быть глобально установлен `bower` и все необходимые зависимости которые описаны в файле `bower.json`:
31 |
32 | ```sh
33 | npm i bower -g
34 | bower i
35 | ```
36 |
37 | Максимальную информацию по опциям Вы можите увидеть в [исходном коде](https://github.com/alx2das/battlefield-js/tree/master/src).
38 |
39 |
40 | #### [Лицензия ISC](https://ru.wikipedia.org/wiki/%D0%9B%D0%B8%D1%86%D0%B5%D0%BD%D0%B7%D0%B8%D1%8F_ISC)
--------------------------------------------------------------------------------
/src/css/battlefield.logger.css:
--------------------------------------------------------------------------------
1 | .bf-logger {
2 | max-width: 950px;
3 | margin: 0 auto;
4 | }
5 |
6 | .bf-logger h2 {
7 | padding-bottom: 15px;
8 | }
9 |
10 | .bf-logger ul {
11 | list-style: none;
12 | max-height: 300px;
13 | overflow-y: auto;
14 | margin-bottom: 25px;
15 | }
16 |
17 | .bf-logger ul li {
18 | display: block;
19 | padding: 7px 0;
20 | border-bottom: 1px solid #d6d6d6;
21 | }
22 |
23 | .bf-logger ul li.no-active {
24 | display: none;
25 | }
26 |
27 | .bf-logger ul li:hover {
28 | background-color: #f6f6f6
29 | }
30 |
31 | .game-log ul li:last-child {
32 | border-bottom: none;
33 | }
34 |
35 | .bf-logger ul li .point, .bf-logger ul li .type, .bf-logger ul li .time {
36 | width: 60px;
37 | display: inline-block;
38 | }
39 |
40 | .bf-logger ul li .point, .bf-logger ul li .time {
41 | font-weight: 700
42 | }
43 |
44 | .bf-logger ul li .point {
45 | }
46 |
47 | .bf-logger ul li .type {
48 | margin: 0 10px;
49 | font-size: 12px;
50 | padding: 2px 0;
51 | text-align: center;
52 | border-radius: 2px;
53 | font-weight: 700;
54 | }
55 |
56 | .bf-logger ul li .type.kil {
57 | background-color: #d9534f;
58 | color: #fff;
59 | }
60 |
61 | .bf-logger ul li .type.war {
62 | background-color: #f0ad4e;
63 | color: #fff;
64 | }
65 |
66 | .bf-logger ul li .type.nul {
67 | }
68 |
69 | .bf-logger ul li .name {
70 | font-size: 14px
71 | }
72 |
73 | .bf-logger ul li .time {
74 | float: right;
75 | }
76 |
77 | .bf-logger .bf-filter {
78 | float: right;
79 | font-weight: 400;
80 | font-size: 13px;
81 | }
82 |
83 | .bf-logger .bf-filter.no-active {
84 | display: none;
85 | }
86 |
87 | .bf-logger .bf-filter span.active {
88 | border-bottom: 1px solid;
89 | }
--------------------------------------------------------------------------------
/src/js/battlefield.options.js:
--------------------------------------------------------------------------------
1 | var options = {}, // глобальный объект с опциями игры
2 | // публичные свойства, могут быть изменены через config
3 | _optionsPublic = {
4 | printName: true, // печатать имена игроков
5 | printLog: true, // ведения журнала боя
6 | printHelp: true, // использовать подсказки
7 |
8 | fSize: { // размерность игрового поля
9 | h: 15, v: 15 // h - горизонтально, v - вертикально
10 | },
11 | fBarrier: [ // список кораблей на игровом поле
12 | [5, 2], [4, 3], [3, 4], [2, 5], [1, 6] // по принципу: ["кол-во палуб", "кол-во штук"]
13 | ],
14 | fMarker: { // обозначение ячеек
15 | h: 'number', v: 'char' // могут быть: char или number
16 | }
17 | },
18 | // приватные, не доступны для изменения
19 | _optionsPrivate = {
20 | player: { // коды играков, должны начинаться с буквы
21 | kUser: 'FUser', kBrain: 'FBrain'
22 | },
23 | winner: {
24 | kUser: 0, kBrain: 0
25 | },
26 | tPoint: { // типы точек игрового поля
27 | DEF: 0, BAR: 1, KIL: 2, NUL: 3 // пусто, корабль, ранил, мимо
28 | }
29 | };
--------------------------------------------------------------------------------
/examples/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Battlefield JS
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
29 |
30 |
31 |
32 |
33 |
34 |
42 |
43 |
46 |
47 |
48 |
52 |
53 |
54 |
55 |
56 |
57 |
--------------------------------------------------------------------------------
/examples/development.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Battlefield JS
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
34 |
35 |
36 |
37 |
38 |
39 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
60 |
61 |
62 |
63 |
64 |
65 |
--------------------------------------------------------------------------------
/src/css/battlefield.global.css:
--------------------------------------------------------------------------------
1 | * {
2 | margin: 0;
3 | padding: 0
4 | }
5 |
6 | html, body {
7 | min-height: 100%;
8 | height: 100%;
9 | font-family: Arial, sans-serif;
10 | }
11 |
12 | .wrap {
13 | width: 950px;
14 | margin: 0 auto;
15 | }
16 |
17 | .container {
18 | min-height: 100%;
19 | }
20 |
21 | .container .clear-footer {
22 | height: 46px;
23 | }
24 |
25 | .ctn-header:after, .ctn-footer:after {
26 | content: " ";
27 | clear: both;
28 | display: block;
29 | }
30 |
31 | .ctn-header {
32 | padding: 20px 0;
33 | border-bottom: 1px solid #7a8b9f;
34 | }
35 |
36 | .ctn-header .logotype {
37 | float: left;
38 | color: #7a8b9f;
39 | font-weight: 700;
40 | font-size: 26px;
41 | text-decoration: none;
42 | }
43 |
44 | .ctn-header .logotype small {
45 | font-weight: 400;
46 | font-size: 12px;
47 | color: #b5b5b5;
48 | }
49 |
50 | .ctn-header .soc-btn {
51 | float: right;
52 | padding: 3px 0;
53 | }
54 |
55 | .ctn-footer {
56 | padding: 15px 0;
57 | margin-top: -46px;
58 | border-top: 1px solid #7a8b9f;
59 | font-size: 13px;
60 | color: #b5b5b5;
61 | }
62 |
63 | .ctn-footer .copy {
64 | float: left;
65 | }
66 |
67 | .ctn-footer .file {
68 | float: left;
69 | margin-left: 25px;
70 | }
71 |
72 | .ctn-footer .file a {
73 | color: #7a8b9f;
74 | text-decoration: none;
75 | }
76 |
77 | .ctn-footer .file a:hover {
78 | border-bottom: 1px solid #7a8b9f
79 | }
80 |
81 | .bf-player-name, .bf-fields, .bf-logger, .bf-modal-window, .btn {
82 | font-family: Arial, sans-serif;
83 | color: #7a8b9f;
84 | font-size: 13px;
85 | }
86 |
87 | .bf-player-name, .bf-fields, .bf-logger {
88 | padding: 25px 0;
89 | }
90 |
91 | .bf-player-name:after, .bf-fields:after, .bf-logger:after {
92 | content: " ";
93 | clear: both;
94 | display: block;
95 | }
96 |
97 | .btn {
98 | color: #fff;
99 | float: left;
100 | margin-right: 10px;
101 | outline: 0;
102 | margin-bottom: 0;
103 | font-weight: 400;
104 | text-align: center;
105 | vertical-align: middle;
106 | cursor: pointer;
107 | border: 1px solid transparent;
108 | white-space: nowrap;
109 | padding: 7px 14px;
110 | font-size: 13px;
111 | line-height: 1.2;
112 | border-radius: 3px;
113 | -webkit-user-select: none;
114 | -moz-user-select: none;
115 | -ms-user-select: none;
116 | user-select: none;
117 | }
118 |
119 | .btn {
120 | background-color: #5cb85c;
121 | color: #fff;
122 | }
123 |
124 | .btn:active {
125 | background-color: #449d44;
126 | }
127 |
128 | .btn:hover {
129 | background-color: #449d44;
130 | }
131 |
132 | .btn.btn-warning {
133 | background-color: #f0ad4e;
134 | }
135 |
136 | .btn.btn-warning:active {
137 | background-color: #ec971f;
138 | }
139 |
140 | .btn.btn-warning:hover {
141 | background-color: #ec971f;
142 | }
143 |
144 | .btn.btn-danger {
145 | background-color: #d9534f;
146 | float: right;
147 | margin-right: 0
148 | }
149 |
150 | .btn.btn-danger:active {
151 | background-color: #c9302c;
152 | background-image: none;
153 | }
154 |
155 | .btn.btn-danger:hover {
156 | background-color: #c9302c;
157 | }
158 |
159 | span.js {
160 | margin-left: 15px;
161 | cursor: pointer;
162 | color: #7a8b9f;
163 | border-bottom: 1px dashed #7a8b9f;
164 | }
165 |
166 | span.js:hover {
167 | border-bottom: none
168 | }
--------------------------------------------------------------------------------
/test/spec/class.battlefield.js:
--------------------------------------------------------------------------------
1 | describe('Публичный класс Battlefield()', function() {
2 | var BF;
3 |
4 | beforeEach(function () {
5 | options = {};
6 | _instances = false;
7 | });
8 | afterEach(function () {
9 | BF = undefined;
10 | });
11 |
12 | it("Применение публичных опций", function() {
13 | BF = new Battlefield('htmlSelector', {
14 | fSize: { h: 22, v: 23 },
15 | fBarrier: [
16 | [6, 1], [5, 2], [4, 3], [3, 3], [2, 4], [1, 6]
17 | ]
18 | });
19 |
20 | expect(options.fSize.h).toEqual(22);
21 | expect(options.fSize.v).toEqual(23);
22 | expect(options.fBarrier.length).toEqual(6);
23 | expect(options.fBarrier[0]).toEqual([6, 1]);
24 | });
25 |
26 | it("Игнорирование изменения приватных опций", function() {
27 | BF = new Battlefield('htmlSelector', {
28 | player: {
29 | kUser: 'BlaBlaUser', kBrain: 'BlaBlaBrain'
30 | },
31 | tPoint: {
32 | DEF: '12', BAR: '13', KIL: '14', NUL: '15'
33 | }
34 | });
35 |
36 | expect(options.player.kUser).not.toBe('BlaBlaUser');
37 | expect(options.player.kBrain).not.toBe('BlaBlaBrain');
38 |
39 | expect(options.tPoint.DEF).not.toBe('12');
40 | expect(options.tPoint.DEF).not.toBe('13');
41 | expect(options.tPoint.DEF).not.toBe('14');
42 | expect(options.tPoint.DEF).not.toBe('15');
43 | });
44 |
45 | it('Установка уровня сложности', function () {
46 | BF = new Battlefield('htmlSelector');
47 |
48 | expect(options.fSize.h).not.toBe(10);
49 | expect(options.fSize.v).not.toBe(10);
50 | expect(options.fBarrier.length).not.toBe(4);
51 |
52 | BF.setLevel('easy');
53 |
54 | expect(options.fSize.h).toBe(10);
55 | expect(options.fSize.v).toBe(10);
56 | expect(options.fBarrier.length).toBe(4);
57 |
58 | BF.setLevel('middle');
59 |
60 | expect(options.fSize.h).toBe(15);
61 | expect(options.fSize.v).toBe(15);
62 | expect(options.fBarrier.length).toBe(5);
63 |
64 | BF.setLevel('hard');
65 |
66 | expect(options.fSize.h).toBe(15);
67 | expect(options.fSize.v).toBe(20);
68 | expect(options.fBarrier.length).toBe(6);
69 | });
70 | });
71 |
72 | describe('Приватный класс Field()', function () {
73 | var BF;
74 | var F;
75 |
76 | beforeEach(function () {
77 | options = {};
78 | _instances = false;
79 |
80 | BF = new Battlefield('htmlSelector');
81 | });
82 | afterEach(function () {
83 | BF = undefined;
84 | });
85 |
86 | it('Создание игровых полей', function () {
87 | BF.setLevel('easy');
88 | F = new Field();
89 | expect(F.fields[options.player.kUser].length).toBe(10);
90 | expect(F.fields[options.player.kUser][0].length).toBe(10);
91 |
92 | delete F;
93 |
94 | BF.setLevel('middle');
95 | F = new Field();
96 | expect(F.fields[options.player.kUser].length).toBe(15);
97 | expect(F.fields[options.player.kUser][0].length).toBe(15);
98 |
99 | delete F;
100 |
101 | BF.setLevel('hard');
102 | F = new Field();
103 | expect(F.fields[options.player.kUser].length).toBe(20);
104 | expect(F.fields[options.player.kUser][0].length).toBe(15);
105 | })
106 | });
--------------------------------------------------------------------------------
/gulpfile.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | var pkg = require('./package.json'),
4 | gulp = require('gulp'),
5 | autoprefixer = require('autoprefixer'),
6 | postcss = require('gulp-postcss'),
7 | concat = require('gulp-concat'),
8 | uglify = require('gulp-uglify'),
9 | minifyCSS = require('gulp-minify-css'),
10 | rename = require("gulp-rename"),
11 | del = require('del'),
12 | runSequence = require('run-sequence');
13 |
14 | // очистка директории
15 | gulp.task('clean', function () {
16 | return del('dist');
17 | });
18 |
19 | // обьединение файлов
20 | gulp.task('concat', function (callback) {
21 | runSequence(
22 | ['concat:js', 'concat:css'],
23 | callback
24 | );
25 | });
26 |
27 | // обьединение JS файлов
28 | gulp.task('concat:js', function () {
29 | return gulp
30 | .src([
31 | 'src/js/battlefield.options.js',
32 | 'src/js/battlefield.class-battlefield.js',
33 | 'src/js/battlefield.class-field.js',
34 | 'src/js/battlefield.class-battle.js',
35 | 'src/js/battlefield.class-gameui.js',
36 | 'src/js/battlefield.helper.js'
37 | ])
38 | .on('data', function (file) {
39 | file.contents = new Buffer(file.contents.toString() + '\n');
40 | })
41 | .pipe(concat('battlefield.js'))
42 | .on('data', function (file) {
43 | file.contents = new Buffer(
44 | description() +
45 | '(function(window){' +
46 | ' \n' +
47 | ' "use strict";' +
48 | ' \n\n' +
49 | file.contents.toString() +
50 | ' \n' +
51 | ' window.Battlefield = Battlefield;' +
52 | ' \n' +
53 | '})(window);'
54 | );
55 | })
56 | .pipe(gulp.dest('dist'));
57 | });
58 |
59 | // обьединение CSS файлов
60 | gulp.task('concat:css', function () {
61 | return gulp
62 | .src('src/css/*.css')
63 | .pipe(postcss([autoprefixer()]))
64 | .pipe(concat('battlefield.css'))
65 | .on('data', function (file) {
66 | file.contents = new Buffer(
67 | description() +
68 | file.contents.toString()
69 | )
70 | })
71 | .pipe(gulp.dest('dist'));
72 | });
73 |
74 | // минификация файлов
75 | gulp.task('min', function (callback) {
76 | runSequence(
77 | ['min:js', 'min:css'],
78 | callback
79 | );
80 | });
81 |
82 | // добавляем метрику
83 | gulp.task('metrika', function () {
84 | return gulp
85 | .src([
86 | 'dist/battlefield.js',
87 | 'src/js/metrika.js'
88 | ])
89 | .on('data', function (file) {
90 | file.contents = new Buffer(file.contents.toString() + '\n');
91 | })
92 | .pipe(concat('battlefield.js'))
93 | .pipe(gulp.dest('dist'));
94 | });
95 |
96 | // минификация JS файлов
97 | gulp.task('min:js', function () {
98 | return gulp
99 | .src(['dist/battlefield.js'])
100 | .pipe(rename('battlefield.min.js'))
101 | .pipe(uglify())
102 | .on('data', function (file) {
103 | file.contents = new Buffer(
104 | description() +
105 | file.contents.toString()
106 | );
107 | })
108 | .pipe(gulp.dest('dist'));
109 | });
110 |
111 | // минификация CSS файлов
112 | gulp.task('min:css', function () {
113 | return gulp
114 | .src(['dist/battlefield.css'])
115 | .pipe(rename('battlefield.min.css'))
116 | .pipe(minifyCSS())
117 | .on('data', function (file) {
118 | file.contents = new Buffer(
119 | description() +
120 | file.contents.toString()
121 | )
122 | })
123 | .pipe(gulp.dest('dist'));
124 | });
125 |
126 | // главная задача
127 | gulp.task('default', function (callback) {
128 | runSequence(
129 | 'clean',
130 | 'concat',
131 | 'metrika',
132 | 'min',
133 | callback
134 | );
135 | });
136 |
137 | // описание для файлов
138 | function description() {
139 | return '' +
140 | '/**\n' +
141 | ' * ' + pkg.name + ' v' + pkg.version + '\n' +
142 | ' * ' + pkg.description + '\n' +
143 | ' * \n' +
144 | ' * repository: ' + pkg.repository.url + '\n' +
145 | ' * bugs: ' + pkg.repository.url + '\n' +
146 | ' * homepage: ' + pkg.homepage + '\n' +
147 | ' * \n' +
148 | ' * Copyright 2017, ' + pkg.author + '\n' +
149 | ' * Date: ' + new Date().toDateString() + '\n' +
150 | ' */\n';
151 | }
--------------------------------------------------------------------------------
/src/js/battlefield.class-field.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Класс для создания игрового поля
3 | *
4 | * Требуется:
5 | * options = {} - глобальный объект с опциями игры
6 | * h = {} - глобальный обьект функций помошников
7 | *
8 | * @constructor
9 | */
10 | function Field() {
11 | if (options.fSize.h < 10 || options.fSize.h > 25 || options.fSize.v < 10 || options.fSize.v > 25)
12 | throw new RangeError(h.getMessage('err_size_field'));
13 |
14 | if (!(options.fBarrier instanceof Array) || options.fBarrier.length < 1)
15 | throw new RangeError(h.getMessage('err_barrier'));
16 |
17 | this.fields = [];
18 | this.fName = [options.player.kUser, options.player.kBrain];
19 |
20 | for (var player in options.player) { // создание игровых полей
21 | var fKey = options.player[player];
22 |
23 | var field = new Array(options.fSize.v);
24 | for (var i = 0; i < options.fSize.v; i++) {
25 | field[i] = new Array(options.fSize.h);
26 |
27 | for (var j = 0; j < options.fSize.h; j++)
28 | field[i][j] = options.tPoint.DEF;
29 | }
30 |
31 | this.fields[fKey] = field;
32 | }
33 | }
34 |
35 | /**
36 | * Устанавливает корабли на игровое поле
37 | *
38 | * @returns {Array}
39 | */
40 | Field.prototype.setBarrier = function () {
41 | var self = this,
42 | maxIteration = this.fName.length * 100;
43 |
44 | this.fName.forEach(function (fKey) { // на каждое игровое поле ставим по отдельности
45 | var field = self.fields[fKey];
46 |
47 | options.fBarrier.forEach(function (ship) { // перебираем типы кораблей
48 | var cell = ship[0], // кол-во палуб
49 | ctn = ship[1]; // кол-во штук
50 |
51 | for (var c = 0; c < ctn; c++) // устанавливаем по одному
52 | field = shipInField(cell, field);
53 | });
54 |
55 | self.fields[fKey] = field;
56 | });
57 |
58 | return this.fields; // игровые поля с кораблями
59 |
60 |
61 | // private method....................
62 | // ..................................
63 |
64 | // установит корабль на игровое поле
65 | function shipInField(cell, field) {
66 | if (maxIteration > 0) maxIteration--; // от зацикливания рекурсии
67 | else throw new RangeError(h.getMessage('err_max_iteration'));
68 |
69 | var sPoint = [], // массив точек для размещения корабля
70 | x = h.rand(0, options.fSize.v - 1), // случайная координата начала установки
71 | y = h.rand(0, options.fSize.h - 1);
72 |
73 | var posHorizontal = h.rand(1, 2) % 2; // случайное направление, горизонтально/вертикально
74 |
75 | for (var i = 0; i < cell; i++) { // перебор палуб
76 | var pX = x, pY = y;
77 |
78 | if (posHorizontal) pX = x + i; // направление корабля
79 | else pY = y + i;
80 |
81 | if (checkPoint(pX, pY, field)) // если точка свободна, сохраняем ее
82 | sPoint.push([pX, pY]);
83 | else return shipInField(cell, field); // перезапуск
84 | }
85 |
86 | // сюда дайдет если только все точки доступны для размещения
87 | sPoint.forEach(function (point) { // точки которые можно нанести на карту
88 | var x = point[0],
89 | y = point[1];
90 |
91 | field[x][y] = options.tPoint.BAR;
92 | });
93 |
94 | return field; // вернем карту с размещенным кораблем
95 | }
96 |
97 | // проверяет выбранную точку + ее окружение
98 | function checkPoint(x, y, field) {
99 | var cP = [[-1, 0], [-1, -1], [0, -1], [1, -1], [1, 0], [1, 1], [0, 1], [-1, 1]],
100 | pX = 0, pY = 0;
101 | var sX = options.fSize.v - 1,
102 | sY = options.fSize.h - 1;
103 |
104 | if (x < 0 || x > sX || y < 0 || y > sY) // исходная точка в рамках игрового поля
105 | return false;
106 |
107 | if (field[x][y] !== options.tPoint.DEF) // точка свободна
108 | return false;
109 |
110 | for (var i = 0; i < cP.length; i++) { // перебор соседних точек
111 | pX = x + cP[i][0];
112 | pY = y + cP[i][1];
113 |
114 | if (pX >= 0 && pX <= sX && pY >= 0 && pY <= sY) { // сеседние точки могут быть за гранью поля
115 | if (field[pX][pY] != options.tPoint.DEF) // но не могут быть заняты другим кораблем
116 | return false;
117 | }
118 | }
119 |
120 | return true; // если все условия проверки пройдены, можно ставить
121 | }
122 | };
--------------------------------------------------------------------------------
/src/css/battlefield.fields.css:
--------------------------------------------------------------------------------
1 | .bf-fields {
2 | background-color: #7a8b9f;
3 | }
4 |
5 | .bf-fields:after {
6 | content: " ";
7 | display: block;
8 | clear: both
9 | }
10 |
11 | .bf-fields .board {
12 | width: 50%
13 | }
14 |
15 | .bf-fields .board.lf {
16 | float: left
17 | }
18 |
19 | .bf-fields .board.rg {
20 | float: right
21 | }
22 |
23 | .bf-fields table.field {
24 | border-collapse: collapse;
25 | margin: 25px;
26 | }
27 |
28 | .bf-fields table.field td, .bf-fields table.field td .box {
29 | border: 2px solid #E5EAFF;
30 | }
31 |
32 | .bf-fields table.field td .box, .bf-fields table.field td .ll, .bf-fields table.field td .mm {
33 | width: 28px;
34 | height: 28px;
35 | line-height: 28px;
36 | text-align: center;
37 | }
38 |
39 | .bf-fields table.field td .ll, .bf-fields table.field td .mm {
40 | position: absolute;
41 | z-index: 1;
42 | font-family: Georgia;
43 | color: #afafaf;
44 | font-weight: 700;
45 | font-size: 12px
46 | }
47 |
48 | .bf-fields table.field td .ll {
49 | left: -28px
50 | }
51 |
52 | .bf-fields table.field td .mm {
53 | top: -28px
54 | }
55 |
56 | .bf-fields table.field td {
57 | position: relative;
58 | }
59 |
60 | .bf-fields table.field td .box {
61 | float: left;
62 | margin: -2px
63 | }
64 |
65 | .bf-fields table.field td .box.null, .bf-fields table.field td .box.auto-null, .bf-fields table.field td .box.kill, .bf-fields .list-let .box.kill {
66 | background: url() 0 0 no-repeat;
67 | }
68 |
69 | .bf-fields table.field td .box.let {
70 | background-color: #E5EAFF
71 | }
72 |
73 | .bf-fields table.field td .box.null {
74 | opacity: 1 !important;
75 | }
76 |
77 | .bf-fields table.field td .box.auto-null {
78 | opacity: .5;
79 | }
80 |
81 | .bf-fields table.field td .box.kill {
82 | background-position: -28px 0;
83 | background-color: #a1b9d6;
84 | }
85 |
86 | .bf-fields table.field td .box.death {
87 | background-color: #deabab
88 | }
89 |
90 | .bf-fields .shot {
91 | position: absolute;
92 | z-index: 1;
93 | width: 28px;
94 | height: 28px;
95 | margin: -2px;
96 | background-color: rgba(255, 249, 0, 0.7);
97 | border: 2px solid #fff900;
98 | }
99 |
100 | .bf-fields .list-let {
101 | margin: 25px 15px 0;
102 | float: left;
103 | }
104 |
105 | .bf-fields .list-let .let {
106 | margin-bottom: 10px
107 | }
108 |
109 | .bf-fields .list-let .let:after {
110 | content: " ";
111 | display: block;
112 | clear: both
113 | }
114 |
115 | .bf-fields .list-let span {
116 | width: 26px;
117 | height: 26px;
118 | line-height: 26px;
119 | margin: 0 10px;
120 | text-align: center;
121 | color: #afafaf;
122 | font-weight: 700;
123 | display: inline-block;
124 | background: 0 0;
125 | border: none;
126 | }
127 |
128 | .bf-fields .list-let span .shot {
129 | width: 24px;
130 | height: 24px;
131 | margin: -27px 0 0 0;
132 | }
133 |
134 | .bf-fields .list-let .box {
135 | float: left;
136 | width: 24px;
137 | height: 24px;
138 | margin-left: -2px;
139 | background-color: rgba(165, 188, 208, .5) !important;
140 | border: 2px solid #E5EAFF;
141 | }
142 |
143 | .bf-fields .list-let .box.kill {
144 | background-position: -28px 0;
145 | }
146 |
147 | .bf-fields .board.rg table.field td .box {
148 | cursor: pointer;
149 | }
150 |
151 | .bf-fields .board.rg table.field td .box:hover {
152 | box-shadow: inset 0 0 5px 2px #ff8d8d;
153 | }
154 |
155 | .bf-fields .board.rg table.field td .box.kill:hover,
156 | .bf-fields .board.rg table.field td .box.let:hover,
157 | .bf-fields .board.rg table.field td .box.null:hover,
158 | .bf-fields .board.rg table.field td .box.auto-null:hover {
159 | box-shadow: none !important;
160 | cursor: default !important;
161 | }
162 |
163 | .bf-fields .board table.field.timeout {
164 | opacity: 0.7;
165 | -webkit-user-select: none;
166 | -moz-user-select: none;
167 | -ms-user-select: none;
168 | user-select: none;
169 | }
170 |
171 | .bf-fields .board table.field.timeout .box {
172 | cursor: wait !important;
173 | }
174 |
175 | .bf-fields .board.lf table.field,
176 | .bf-fields .board.lf .list-let,
177 | .bf-fields .board.lf .list-let .box,
178 | .bf-fields .board.rg .list-let span {
179 | float: right
180 | }
181 |
182 | .bf-fields .board.rg table.field,
183 | .bf-fields .board.rg .list-let,
184 | .bf-fields .board.rg .list-let .box,
185 | .bf-fields .board.lf .list-let span {
186 | float: left
187 | }
--------------------------------------------------------------------------------
/src/js/battlefield.helper.js:
--------------------------------------------------------------------------------
1 | // глобальный класс модуля
2 | var h = {
3 | /**
4 | * имена играков по умолчанию
5 | */
6 | playerName: {
7 | def: {
8 | kUser: 'Игрок', kBrain: 'Компьютер'
9 | }
10 | },
11 |
12 | /**
13 | * Вернет случайное число в интервале
14 | *
15 | * @param min - по умолчанию: 0
16 | * @param max - по умолчанию: 100
17 | * @returns {Number}
18 | */
19 | rand: function (min, max) {
20 | min = min || 0;
21 | max = max || 100;
22 |
23 | return parseInt(Math.random() * (max - min + 1) + min);
24 | },
25 |
26 | /**
27 | * Вернет буквку англ.алфавита по номеру.
28 | * Если номер больше кол-ва букв, к результату будет прибавлена цифра
29 | *
30 | * @param key
31 | * @param operand
32 | * @returns {*}
33 | */
34 | getLetter: function (key, operand) {
35 | var operand = operand || '',
36 | alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
37 | if (key > alphabet.length)
38 | return this.getLetter(key - alphabet.length, (operand == '' ? 1 : operand + 1));
39 |
40 | return alphabet[key] + operand;
41 | },
42 |
43 | /**
44 | * Перемешает массив случайным образом
45 | *
46 | * @param arr - массив
47 | * @returns {*}
48 | */
49 | shuffle: function (arr) {
50 | var inx, buffer;
51 | for (var i = 0; i < arr.length - 1; i++) {
52 | inx = this.rand(0, arr.length - 1);
53 | buffer = arr[inx];
54 |
55 | arr[inx] = arr[arr.length - 1];
56 | arr[arr.length - 1] = buffer;
57 | }
58 | return arr;
59 | },
60 |
61 | /**
62 | * Интернализация
63 | *
64 | * @param type
65 | * @returns {*}
66 | */
67 | getMessage: function (type) {
68 | var mess = {
69 | start_game: 'Начать игру',
70 | new_game: 'Новая игра',
71 | game_over: 'Игра закончена',
72 | you_lose: 'Ты проиграл',
73 | you_winner: 'Ты выиграл',
74 | options: 'Настройки',
75 | update_options: 'Изменить настройки',
76 | close: 'Закрыть',
77 | update_page: 'Обновить страницу',
78 | set_default_params: 'Настройки по умолчанию',
79 | save: 'Сохранить',
80 | filter: 'Фильтр',
81 | full: 'Все',
82 |
83 | log_kill: 'ранил',
84 | log_death: 'убил',
85 | log_past: 'мимо',
86 |
87 | // error
88 | err_invalid_selector: 'Указанный HTML блок на странице не найден',
89 | err_max_iteration: 'Не получилось установить корабли на игровое поле, измените настройки',
90 | err_invalid_player: 'Неизвестный игрок',
91 | err_invalid_field: 'Некорректное игровое поле',
92 | err_player_name: 'Переданны не корректные ключи играков',
93 | err_size_field: 'Игровое поле не должно быть меньше 10х10 или больше 25х25, измените настройки',
94 | err_barrier: 'Не корректный список кораблей',
95 | err_invalid_level: 'Указан несуществующий уровень сложности',
96 |
97 | // info
98 | info_log_title: 'Состояние игрового поля',
99 | info_title_error: 'Критическая ошибка',
100 | info_title_range_error: 'Некорректные настройки',
101 | info_winner: 'Замечательная победа,
попробуешь еще?',
102 | info_loser: 'В следующий раз повезет больше,
попробуешь еще?'
103 | };
104 |
105 | return typeof mess[type] == 'string' && mess[type].length > 0 ? mess[type] : type;
106 | },
107 |
108 | /**
109 | * Вернет строковое имя игрока
110 | *
111 | * @param fKey - строковый ключ игрока
112 | * @returns {*}
113 | */
114 | getPlayerName: function (fKey) {
115 | var key = fKey == options.player.kUser ? 'kUser' : 'kBrain';
116 | return this.playerName.def[key];
117 | },
118 |
119 | /**
120 | * Ловит сообщение об ошибке
121 | *
122 | * @param err
123 | */
124 | showExceptions: function (err) {
125 | var title = '',
126 | content = '',
127 | button = [];
128 |
129 | switch (err.name) {
130 | case ('Error'):
131 | title = h.getMessage('info_title_error');
132 | content = err.message;
133 | button = [{
134 | elValue: h.getMessage('set_default_params'),
135 | onClick: function (modal) {
136 | modal.close();
137 | var b = new Battlefield(options.htmlSelector, options);
138 | b.setLevel('middle').run();
139 | }
140 | }];
141 | break;
142 | case ('RangeError'):
143 | title = h.getMessage('info_title_range_error');
144 | content = err.message;
145 | button = [{
146 | elValue: h.getMessage('update_options'),
147 | onClick: function (modal) {
148 | modal.close();
149 | var b = new Battlefield(options.htmlSelector, options);
150 | b.updateConfig();
151 | }
152 | },{
153 | elValue: h.getMessage('set_default_params'),
154 | elClass: 'btn-warning',
155 | onClick: function (modal) {
156 | modal.close();
157 | var b = new Battlefield(options.htmlSelector, options);
158 | b.setLevel('middle').run();
159 | }
160 | }];
161 | break;
162 | default:
163 | title = err.name;
164 | content = err.message;
165 | }
166 |
167 | this.modalWindow(title, content, button);
168 | },
169 |
170 | /**
171 | * Выводит модальное окно, автоматически установит по середине страницы
172 | *
173 | * ~~~~
174 | * h.modalWindow(
175 | * 'title modal window',
176 | * 'Content modal',
177 | * [{
178 | * elClass: 'btn-success', // css класс будет добавлен к кнопке
179 | * elValue: 'Button close', // значение кнопки
180 | * onClick: function (modal, event) {} // событие клика по кнопке
181 | * }]
182 | * )
183 | * ~~~~
184 | *
185 | * @param title - заголовок
186 | * @param contentHTML - html содержимое
187 | * @param button - массив обьектов кнопок модального окна
188 | */
189 | modalWindow: function (title, contentHTML, button) {
190 | var modal = {
191 | font: false,
192 | window: false,
193 |
194 | construct: function () {
195 | this.createHTML().autoPosition();
196 |
197 | if (!(button instanceof Array) || button.length == 0) {
198 | this.window.querySelector('.footer').style.display = 'none';
199 | return false;
200 | }
201 |
202 | var footer = this.window.querySelector('.footer'),
203 | on = [];
204 |
205 | for (var i = 0; i < button.length; i++) {
206 | var btn = button[i],
207 | dopClass = typeof button[i].elClass == 'string' ? ' ' + btn.elClass : '';
208 |
209 | on[i] = document.createElement('button');
210 | on[i].setAttribute('class', 'btn' + dopClass);
211 | on[i].setAttribute('id', 'btn_' + i);
212 | on[i].innerHTML = btn.elValue;
213 | on[i].addEventListener('click', eventListener(i, button), false);
214 |
215 | footer.appendChild(on[i]);
216 | }
217 | },
218 | createHTML: function () {
219 | var _font = document.querySelector('.bf-modal-font'),
220 | _window = document.querySelector('.bf-modal-window');
221 | if (_font != null) document.body.removeChild(_font);
222 | if (_window != null) document.body.removeChild(_window);
223 |
224 |
225 | this.font = document.createElement('div');
226 | this.font.setAttribute('class', 'bf-modal-font');
227 |
228 | this.window = document.createElement('div');
229 | this.window.setAttribute('class', 'bf-modal-window');
230 | this.window.innerHTML =
231 | '' +
232 | ' ' +
233 | '
' + contentHTML + '
' +
234 | ' ' +
235 | '
';
236 |
237 | document.body.appendChild(this.font);
238 | document.body.appendChild(this.window);
239 |
240 | return this;
241 | },
242 | autoPosition: function () {
243 | var wid2 = this.window.clientWidth / 2,
244 | hei2 = this.window.clientHeight / 2;
245 |
246 | this.window.style.marginTop = hei2 * (-1) + 'px';
247 | this.window.style.marginLeft = wid2 * (-1) + 'px';
248 | },
249 | close: function () {
250 | document.body.removeChild(this.font);
251 | document.body.removeChild(this.window);
252 | }
253 | };
254 |
255 | modal.construct();
256 |
257 |
258 | // private method....................
259 | // ..................................
260 |
261 | // вешает событие на кнопку в модальном окне
262 | function eventListener(i, obj) {
263 | return function (event) {
264 | if (typeof obj[i].onClick == 'function')
265 | obj[i].onClick(modal, event);
266 | };
267 | }
268 | },
269 |
270 | /**
271 | * Запускает анимацию плавного исчезновения со страницы
272 | *
273 | * @param element - document элемент
274 | * @param time - кол-во сек.через которое элемент исчезнет со страницы
275 | */
276 | animateOpacity: function (element, time) {
277 | element.style.opacity = 1;
278 | var t = setInterval(function () {
279 | element.style.opacity = element.style.opacity - (100 / (time / 0.1));
280 | if (element.style.opacity <= 0) {
281 | clearInterval(t);
282 | try {
283 | element.parentNode.removeChild(element);
284 | } catch (e) { }
285 | }
286 | }, 1);
287 | }
288 | };
--------------------------------------------------------------------------------
/src/js/battlefield.class-battlefield.js:
--------------------------------------------------------------------------------
1 | var _instances = false, // Singleton
2 | _globalLevel = 'middle'; // уровень сложности по умолчанию
3 |
4 | /**
5 | * Инициализирует игру, устанавливает параметры по умолчанию
6 | *
7 | * Требуется:
8 | * options = {} - глобальный объект с опциями игры
9 | * h = {} - глобальный обьект функций помошников
10 | *
11 | * @param htmlSelector - строковый селектор элемента на странице
12 | * @param config - объект настроек по умолчанию, доступные параметры: _optionsPublic
13 | * @returns {Battlefield}
14 | * @constructor
15 | */
16 | function Battlefield(htmlSelector, config) {
17 | // Singleton
18 | if (_instances instanceof Battlefield)
19 | return _instances;
20 | _instances = this;
21 |
22 | // объединяем установленные опции игры с доступными
23 | if (typeof config == 'object') {
24 | for (var key in _optionsPublic)
25 | options[key] = config.hasOwnProperty(key) ? config[key] : _optionsPublic[key];
26 | options = Object.assign(options, _optionsPrivate);
27 | }
28 | else options = Object.assign(_optionsPublic, _optionsPrivate);
29 |
30 | options.htmlSelector = document.querySelector(htmlSelector);
31 | }
32 |
33 | /**
34 | * Запуск игры
35 | */
36 | Battlefield.prototype.run = function () {
37 | try {
38 | var F = new Field();
39 | var fields = F.setBarrier();
40 |
41 | new Battle(fields, this);
42 | } catch (err) {
43 | h.showExceptions(err);
44 | }
45 | };
46 |
47 | /**
48 | * Вернет настройки для указанного уровеня сложности
49 | *
50 | * @param type
51 | * @returns {*}
52 | */
53 | Battlefield.prototype.getLevel = function (type) {
54 | var level = {
55 | easy: {
56 | fSize: {h: 10, v: 10},
57 | fBarrier: [[4, 1], [3, 2], [2, 3], [1, 4]]
58 | },
59 | middle: {
60 | fSize: {h: 15, v: 15},
61 | fBarrier: [[5, 2], [4, 3], [3, 4], [2, 5], [1, 6]]
62 | },
63 | hard: {
64 | fSize: {h: 15, v: 20},
65 | fBarrier: [[6, 1], [5, 2], [4, 3], [3, 4], [2, 5], [1, 6]]
66 | }
67 | };
68 |
69 | if (typeof level[type] == 'object') {
70 | return level[type];
71 | }
72 | else throw new Error(h.getMessage('err_invalid_level'));
73 | };
74 |
75 | /**
76 | * Установит уровень сложности
77 | *
78 | * @param type
79 | * @returns {Battlefield}
80 | */
81 | Battlefield.prototype.setLevel = function (type) {
82 | var nLevel = this.getLevel(type);
83 |
84 | options.fSize = nLevel.fSize;
85 | options.fBarrier = nLevel.fBarrier;
86 |
87 | _globalLevel = type;
88 |
89 | return this;
90 | };
91 |
92 | /**
93 | * Выведет модальное окно для изменения настроек игры
94 | */
95 | Battlefield.prototype.updateConfig = function () {
96 | var self = this;
97 |
98 | var contentHtml =
99 | '';
132 | contentHtml = '';
133 |
134 | var newLevel = _globalLevel;
135 |
136 | h.modalWindow(h.getMessage('options'), contentHtml, [{
137 | elValue: h.getMessage('new_game'),
138 | onClick: function (modal) {
139 | modal.close();
140 |
141 | try {
142 | if (newLevel == 'user') {
143 | var nOpt = getUserOptions();
144 |
145 | options.fSize = nOpt.fSize;
146 | options.fBarrier = nOpt.fBarrier;
147 |
148 | _globalLevel = newLevel;
149 | } else {
150 | if (newLevel.length > 0 && newLevel != _globalLevel)
151 | self.setLevel(newLevel);
152 | }
153 |
154 | self.run();
155 | } catch (err) {
156 | h.showExceptions(err);
157 | }
158 | }
159 | },{
160 | elValue: h.getMessage('set_default_params'),
161 | elClass: 'btn-warning',
162 | onClick: function (modal) {
163 | modal.close();
164 |
165 | self.setLevel('middle');
166 | self.run();
167 | }
168 | },{
169 | elValue: h.getMessage('close'),
170 | elClass: 'btn-danger',
171 | onClick: function (modal) {
172 | modal.close();
173 | }
174 | }
175 | ]);
176 |
177 | var table = document.querySelector('.bf-config');
178 | var btnList = document
179 | .getElementById('config_level')
180 | .querySelectorAll('button');
181 |
182 | activeFormOptions(_globalLevel == 'user');
183 | echoOptions(options);
184 |
185 | // выбор уровня сложности
186 | btnList.forEach(function (btn) {
187 | btn.onclick = function (env) {
188 |
189 | newLevel = env.target.value;
190 |
191 | btnList.forEach(function (b) {
192 | b.classList.remove('active');
193 | });
194 | env.target.classList += ' active';
195 |
196 | if (newLevel == 'user') {
197 | activeFormOptions(true);
198 | } else {
199 | activeFormOptions(false);
200 | echoOptions(self.getLevel(newLevel));
201 | }
202 |
203 | return false;
204 | }
205 | });
206 |
207 | // private method....................
208 | // ..................................
209 |
210 | // активирует форму для ввода пользовательских настроек
211 | function activeFormOptions(active) {
212 | // размер игрового поля
213 | table.querySelector('#opt-fsize').querySelectorAll('label')
214 | .forEach(function (label) {
215 | var input = label.querySelector('input');
216 |
217 | if (active) {
218 | label.classList = '';
219 | input.disabled = false;
220 | } else {
221 | label.classList = 'no-act';
222 | input.disabled = true;
223 | }
224 | });
225 |
226 | // корабли на поле
227 | table.querySelector('#opt-fbarrier').querySelectorAll('label')
228 | .forEach(function (label) {
229 | var input = label.querySelector('input');
230 |
231 | if (active) {
232 | label.classList = '';
233 | input.disabled = false;
234 | } else {
235 | label.classList = 'no-act';
236 | input.disabled = true;
237 | }
238 | });
239 | }
240 |
241 | // вернет настройки пользователя
242 | function getUserOptions() {
243 | var nOptions = {
244 | fSize: {h: 0, v: 0},
245 | fBarrier: []
246 | };
247 |
248 | // размер игрового поля
249 | nOptions.fSize.h = parseInt(table.querySelector('input#type-h').value);
250 | nOptions.fSize.v = parseInt(table.querySelector('input#type-v').value);
251 |
252 | // корабли на игровом поле
253 | table.querySelector('#opt-fbarrier').querySelectorAll('label')
254 | .forEach(function (label) {
255 | var input = label.querySelector('input'),
256 | ship = parseInt(input.getAttribute('bar')),
257 | ctn = parseInt(input.value);
258 |
259 | if (ctn > 0)
260 | nOptions.fBarrier.push([ship, ctn]);
261 | });
262 |
263 | return nOptions;
264 | }
265 |
266 | // напечатает переданные настройки
267 | function echoOptions(nOptions) {
268 | // размер игрового поля
269 | table.querySelector('input#type-h').value = nOptions.fSize.h;
270 | table.querySelector('input#type-v').value = nOptions.fSize.v;
271 |
272 | // корабли на игровом поле
273 | table.querySelector('#opt-fbarrier').querySelectorAll('input').forEach(function (input) {
274 | input.value = 0;
275 | });
276 |
277 | nOptions.fBarrier.forEach(function (barrier) {
278 | var ship = barrier[0], // кол-во палуб корабля
279 | ctn = barrier[1]; // кол-во кораблей
280 |
281 | table.querySelector('#opt-fbarrier').querySelector('input#bar_' + ship).value = ctn;
282 | });
283 | }
284 | };
--------------------------------------------------------------------------------
/src/js/battlefield.class-battle.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Запускает алгоритмы обстрела игроками поля боя
3 | *
4 | * Требуется:
5 | * options = {} - глобальный объект с опциями игры
6 | * h = {} - глобальный обьект функций помошников
7 | *
8 | * @param fields - обьект массивов игровых поле
9 | * @param battlefield - экземпляр Battlefield
10 | * @constructor
11 | */
12 | function Battle(fields, battlefield) {
13 | if (!(battlefield instanceof Battlefield))
14 | throw new Error('Invalid Battlefield object in the construct Battle');
15 |
16 | GameUI.apply(this, arguments);
17 |
18 | this.fields = fields;
19 | this.battlefield = battlefield;
20 |
21 | this._lsKill = []; // последнее попадание компьютером
22 | this._lsShot = []; // список точек доступных для выстрела
23 | this._ctnCell = []; // остаток кораблей у игрока
24 |
25 | for (var p in options.player) { // заполнение данных
26 | var v = options.player[p];
27 |
28 | this._lsKill[v] = [];
29 | this._lsShot[v] = getListPoint();
30 | this._ctnCell[v] = getCountCell();
31 | }
32 | this.userKey = options.player.kUser; // ключ игрока
33 | this.enemyKey = options.player.kBrain; // ключ компьютера
34 |
35 | // первый ход
36 | this.player = h.rand(1, 2) % 2 ? this.userKey : this.enemyKey;
37 | this.game(this.player);
38 |
39 |
40 | // private method....................
41 | // ..................................
42 |
43 | function getListPoint() {
44 | var __fullPoint = [];
45 | for (var i = 0; i < options.fSize.h; i++) {
46 | for (var j = 0; j < options.fSize.v; j++)
47 | __fullPoint.push([j, i]);
48 | }
49 | return __fullPoint;
50 | }
51 |
52 | function getCountCell() {
53 | var __countCell = 0;
54 | options.fBarrier.forEach(function (ship) {
55 | __countCell += ship[0] * ship[1];
56 | });
57 | return __countCell;
58 | }
59 | }
60 |
61 | Battle.prototype = Object.create(GameUI.prototype); // наследование
62 |
63 | /**
64 | * Осуществляет переход хода
65 | *
66 | * @param fKey - строковый ключ игрока
67 | */
68 | Battle.prototype.game = function (fKey) {
69 | try {
70 | this.showProgress(fKey);
71 | this.player = fKey;
72 |
73 | switch (fKey) {
74 | case (this.userKey):
75 | this.shotUser();
76 | break;
77 | case (this.enemyKey):
78 | this.shotAI(this.userKey);
79 | break;
80 | default:
81 | throw new Error(h.getMessage('err_invalid_player'));
82 | }
83 | } catch (err) {
84 | h.showExceptions(err);
85 | }
86 | };
87 |
88 | /**
89 | * Ход игрока,
90 | * ожидает событие клика по игровому полю
91 | */
92 | Battle.prototype.shotUser = function () {
93 | var self = this;
94 |
95 | this.clickToField(this.enemyKey, function (event) {
96 | if (self.player !== self.userKey)
97 | return false;
98 |
99 | try {
100 | var Y = event.target.parentNode.cellIndex,
101 | X = event.target.parentNode.parentNode.rowIndex;
102 |
103 | self.shot([X, Y], self.enemyKey);
104 | } catch (err) {
105 | self.game(self.userKey);
106 | }
107 | });
108 | };
109 |
110 | /**
111 | * Ход компьютера,
112 | * в зависимости от попадания по кораблю противника будет
113 | * добивать корабль или стрелять по случайной клетке
114 | *
115 | * @param fKey - строковый ключ игрока
116 | */
117 | Battle.prototype.shotAI = function (fKey) {
118 | var self = this,
119 | field = this.fields[fKey],
120 | fSize = options.fSize,
121 | tPoint = options.tPoint,
122 | point = getPointShot();
123 |
124 | setTimeout(function () {
125 | self.shot(point, fKey);
126 | }, 600);
127 |
128 |
129 | // private method....................
130 | // ..................................
131 |
132 | // вернет случайную точку для выстрела
133 | function getPointShot() {
134 | var point = [];
135 |
136 | if (self._lsKill[fKey].length > 0) { // если ранее попадали по кораблю
137 | point = getPointFinishing();
138 | } else { // случайный выстрел
139 | var rInx = h.rand(0, self._lsShot[fKey].length - 1);
140 | point = self._lsShot[fKey][rInx]; // получаем случайную точку
141 | self._lsShot[fKey].splice(rInx, 1); // удаляем точку из доступных для выстрела
142 | }
143 |
144 | // проверка полученной точки...
145 | if (point instanceof Array && point[0] !== 'undefined' && point[1] !== 'undefined') {
146 | var X = point[0], Y = point[1];
147 | if (X >= 0 && X < options.fSize.v && Y >= 0 && Y < options.fSize.h) {
148 | if (field[X][Y] == tPoint.NUL || field[X][Y] == tPoint.KIL) {
149 | // уже стреляли в эту точку
150 | }
151 | else return point;
152 | }
153 | }
154 |
155 | return getPointShot();
156 | }
157 |
158 | // вернет точку для добивания
159 | function getPointFinishing() {
160 | var cP = [[-1, 0], [0, -1], [1, 0], [0, 1]], // возможные направления для добивания
161 | lsP = self._lsKill[fKey],
162 | point = [];
163 |
164 | var i = 0, n = 0;
165 |
166 | if (lsP.length == 1) { // первый выстрел при добивание
167 | var X = lsP[0][0], Y = lsP[0][1];
168 |
169 | cP = h.shuffle(cP); // перемешает массив
170 | for (i = 0; i < cP.length; i++) { // выбираем случайное направление для выстрела
171 | var pX = X + cP[i][0],
172 | pY = Y + cP[i][1];
173 |
174 | // точка в рамках поля и туда еще не стреляли
175 | if (pX >= 0 && pX < fSize.v && pY >= 0 && pY < fSize.h) {
176 | if (field[pX][pY] !== tPoint.NUL)
177 | return [pX, pY];
178 | }
179 | }
180 | } else { // второй и т.д выстрел при добивании
181 | var posHorizontal = lsP[0][0] == lsP[1][0], // определяем направление корабля
182 | min = fSize.h + fSize.v, max = 0; // крайние точки
183 |
184 | for (i = 0; i < lsP.length; i++) { // поиск крайних точек корабля
185 | n = posHorizontal ? lsP[i][1] : lsP[i][0];
186 |
187 | min = min > n ? n : min;
188 | max = max < n ? n : max;
189 | }
190 |
191 | // случайным образом определяем с какого края стрелять по караблю
192 | var nP = h.rand(1, 2) % 2 ? min - 1 : max + 1;
193 | point = posHorizontal ? [lsP[0][0], nP] : [nP, lsP[0][1]];
194 | }
195 |
196 | return point;
197 | }
198 | };
199 |
200 | /**
201 | * Осуществляет выстрел по игровому полю
202 | *
203 | * @param point - массив 2 элементов с точкой на игровом поле
204 | * @param fKey - строковый ключ игрока
205 | */
206 | Battle.prototype.shot = function (point, fKey) {
207 | var self = this,
208 | ctnCellShip = 0,
209 | _fKey = fKey == this.userKey ? this.enemyKey : this.userKey;
210 | var X = point[0], Y = point[1];
211 |
212 | var check = checkPoint(point, fKey); // проверка выстрела
213 | if (typeof check == 'boolean') { // во что то попал
214 | if (check) { // попал на по кораблю -> РАНИЛ
215 | this._ctnCell[fKey]--; // минус 1 точка для выстрела
216 |
217 | this.fields[fKey][X][Y] = options.tPoint.KIL;
218 | var isKill = isKillShip(point, fKey);
219 | if (typeof isKill == 'boolean') { // корабль просто ранен
220 | this._lsKill[fKey].push(point); // сохраняем последнее попадание
221 |
222 | this.setMarker(point, fKey, options.tPoint.KIL);
223 | this.setHelpMarker(point, fKey, function (_point) {// при ранении установим точки подсказки
224 | var _X = _point[0],
225 | _Y = _point[1];
226 | self.fields[fKey][_X][_Y] = options.tPoint.NUL;
227 | });
228 | }
229 | else { // корабль убит
230 | this._lsKill[fKey] = []; // очищаем список последнего попадания
231 |
232 | isKill.forEach(function (_point) {
233 | var _X = _point[0],
234 | _Y = _point[1];
235 |
236 | // ставим метки на игровое поле, если разрешены подсказки
237 | if (options.printHelp) {
238 | self.setMarker(_point, fKey, options.tPoint.NUL, true);
239 | self.fields[fKey][_X][_Y] = options.tPoint.NUL;
240 | }
241 | if (_fKey == options.player.kBrain)
242 | self.fields[fKey][_X][_Y] = options.tPoint.NUL;
243 | });
244 |
245 | this.setMarker(point, fKey, options.tPoint.KIL + '_death');
246 | this.shipInfoMap(ctnCellShip, fKey);
247 | }
248 |
249 | if (this._ctnCell[fKey] === 0) { // конец игры
250 | this.player = false;
251 | this.gameOver(_fKey);
252 | } else this.game(_fKey); // повтор хода
253 | }
254 | else { // попадание по пустой клетке -> МИМО
255 | this.fields[fKey][X][Y] = options.tPoint.NUL;
256 | this.setMarker(point, fKey, options.tPoint.NUL);
257 | this.game(fKey); // переход кода
258 | }
259 | }
260 | else this.game(_fKey); // уже стрелял в эту точку -> повтор хода
261 |
262 |
263 | // private method....................
264 | // ..................................
265 |
266 | // проверяет точку выстрела по полю
267 | function checkPoint(point, fKey) {
268 | var X = point[0], Y = point[1];
269 |
270 | switch (self.fields[fKey][X][Y]) {
271 | case (options.tPoint.DEF): // мимо
272 | return false;
273 | case (options.tPoint.BAR): // ранил
274 | return true;
275 | default: // что то другое
276 | return null;
277 | }
278 | }
279 |
280 | // проверка уничтожения корабля
281 | function isKillShip(point, fKey) {
282 | var cP = [[-1, 0], [-1, -1], [0, -1], [1, -1], [1, 0], [1, 1], [0, 1], [-1, 1]],
283 | field = self.fields[fKey],
284 | sPoints = [],
285 | lsCellShip = [];
286 |
287 | if (_checkPointToKill(point)) {
288 | lsCellShip.push(point);
289 | ctnCellShip = lsCellShip;
290 |
291 | return sPoints;
292 | }
293 | else return false;
294 |
295 |
296 | // private method....................
297 | // ..................................
298 |
299 | // убит или нет корабль, если да то заполнет массив соседних точек
300 | function _checkPointToKill(point, noCheck) {
301 | var X = point[0], Y = point[1], // исходная проверяемая точка
302 | nX = false, nY = false; // не проверяемая точка при повторе
303 | var dopCheck = []; // дополнительные точки проверки
304 |
305 | if (typeof noCheck == "object") { // не проверяемая точка, при рекурсии
306 | nX = noCheck[0];
307 | nY = noCheck[1];
308 | }
309 |
310 | for (var i = 0; i < cP.length; i++) { // перебор точек по кругу
311 | var pX = X + cP[i][0],
312 | pY = Y + cP[i][1];
313 |
314 | if (pX === nX && pY === nY) { // эту точку уже проверили
315 | // точка проверенна при предыдущей итерации
316 | } else {
317 | // только если точка в рамках игрового поля
318 | if (pX >= 0 && pX < options.fSize.v && pY >= 0 && pY < options.fSize.h) {
319 | var val = field[pX][pY];
320 |
321 | if (val == options.tPoint.BAR)
322 | return false; // целая часть корабля -> не убит
323 | else if (val == options.tPoint.KIL) { // раненая часть корабля, запомним для рекурсии
324 | lsCellShip.push([pX, pY]); // подсчет палуб
325 | dopCheck.push([[pX, pY], [X, Y]]); // доп.точка првоерки
326 | }
327 | else sPoints.push([pX, pY]); // подбитая часть корабля,
328 | }
329 | }
330 | }
331 |
332 | // доп.точки проверки, если еще есть то запуск рекурсии
333 | if (dopCheck.length === 0)
334 | return true; // точек более нет -> убит
335 | else {
336 | var sRes = 0;
337 | for (var k = 0; k < dopCheck.length; k++) {
338 | if (!_checkPointToKill(dopCheck[k][0], dopCheck[k][1]))
339 | return false;
340 | else sRes++;
341 | }
342 | return sRes == dopCheck.length; // проверенно == доп.точек
343 | }
344 | }
345 | }
346 | };
--------------------------------------------------------------------------------
/src/js/battlefield.class-gameui.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Класс управления пользовательским интерфейсом
3 | *
4 | * Требуется:
5 | * options = {} - глобальный объект с опциями игры
6 | * h = {} - глобальный обьект функций помошников
7 | *
8 | * @param htmlSelector - document.querySelector
9 | * @constructor
10 | */
11 | function GameUI(fields, battlefield) {
12 | if (options.htmlSelector == null)
13 | throw new Error(h.getMessage('err_invalid_selector'));
14 |
15 | this.fields = fields;
16 | this.battlefield = battlefield;
17 |
18 | var self = this;
19 | var playerName =
20 | '' +
21 | ' ' + h.getMessage('new_game') + '' +
22 | ' ' + h.getMessage('options') + '' +
23 | '
';
24 |
25 | // очистка блока
26 | options.htmlSelector.innerHTML = '';
27 |
28 | // имена игроков
29 | if (options.printName) {
30 | var echoTotal = options.winner.kUser + options.winner.kBrain != 0;
31 | playerName +=
32 | '' +
33 | '' + h.getPlayerName(options.player.kUser) + '' +
34 | (echoTotal ? '' + options.winner.kUser + '' : '') +
35 | '&' +
36 | (echoTotal ? '' + options.winner.kBrain + '' : '') +
37 | ' ' + h.getPlayerName(options.player.kBrain) + '' +
38 | '
';
39 | }
40 |
41 | this.nameHtml = document.createElement('div');
42 | this.nameHtml.setAttribute('class', 'bf-player-name');
43 | this.nameHtml.innerHTML = playerName;
44 | options.htmlSelector.appendChild(this.nameHtml);
45 |
46 | // игровые поля
47 | this.fieldHtml = document.createElement('div');
48 | this.fieldHtml.setAttribute('class', 'bf-fields');
49 | this.fieldHtml.innerHTML = this.getFieldHTML(options.player);
50 | options.htmlSelector.appendChild(this.fieldHtml);
51 |
52 | // логирование боя
53 | if (options.printLog) {
54 | var filter =
55 | '' +
56 | ' ' + h.getMessage('filter') + ':' +
57 | ' ' + h.getPlayerName(options.player.kUser) + '' +
58 | ' ' + h.getPlayerName(options.player.kBrain) + '' +
59 | ' ' + h.getMessage('full') + '' +
60 | '
';
61 |
62 | this.logFilter = null;
63 | this.actFilter = '';
64 |
65 | this.logHtml = document.createElement('div');
66 | this.logHtml.setAttribute('class', 'bf-logger');
67 | this.logHtml.innerHTML = '' + h.getMessage('info_log_title') + filter + '
';
68 | options.htmlSelector.appendChild(this.logHtml);
69 | }
70 |
71 |
72 | document.getElementById('new_game').onclick = function () {
73 | self.battlefield.run();
74 | };
75 | document.getElementById('config').onclick = function () {
76 | self.battlefield.updateConfig();
77 | };
78 |
79 | }
80 |
81 | /**
82 | * Вернет HTML разметку игрового поля
83 | *
84 | * @param player - строковый ключ игрока
85 | * @returns {string}
86 | */
87 | GameUI.prototype.getFieldHTML = function (player) {
88 | var fields = this.fields,
89 | html = '';
90 |
91 | for (var k in player) {
92 | var fKey = player[k],
93 | dopAttr = fKey == options.player.kUser ? 'lf' : 'rg',
94 | printShip = dopAttr == 'lf';
95 |
96 | html += '';
97 | html += getFieldTable(fKey, printShip);
98 | html += getBarrierInfo(fKey);
99 | html += '
';
100 | }
101 |
102 | return html;
103 |
104 |
105 | // private method....................
106 | // ..................................
107 |
108 | // вернет HTML игрового поля
109 | function getFieldTable(fKey, printShip) {
110 | printShip = printShip || false;
111 | var table = '';
112 |
113 | table += '';
114 | for (var x = 0; x < options.fSize.v; x++) {
115 | table += '';
116 | for (var y = 0; y < options.fSize.h; y++) {
117 | var mm = '', ll = '';
118 |
119 | // установка маркеров
120 | if (options.fMarker !== false && typeof options.fMarker == 'object') {
121 | var txtMM = typeof options.fMarker.h != 'undefined' ? (options.fMarker.h == 'char' ? h.getLetter(y) : (y + 1)) : (y + 1),
122 | txtLL = typeof options.fMarker.v != 'undefined' ? (options.fMarker.v == 'char' ? h.getLetter(x) : (x + 1)) : (x + 1);
123 |
124 | mm = x === 0 ? '' + txtMM + '
' : '';
125 | ll = y === 0 ? '' + txtLL + '
' : '';
126 | }
127 |
128 | var ship = '';
129 | if (printShip)
130 | ship = fields[fKey][x][y] == options.tPoint.BAR ? ' let' : '';
131 | table += '' + mm + ll + ' | ';
132 | }
133 | table += '
';
134 | }
135 | table += '
';
136 |
137 | return table;
138 | }
139 |
140 | // вернет HTML списка кораблей
141 | function getBarrierInfo(fKey) {
142 | var html = '';
143 |
144 | options.fBarrier.forEach(function (ship) {
145 | var box = '';
146 | for (var c = 0; c < ship[0]; c++)
147 | box += '';
148 | html +=
149 | '' +
150 | ' x' + ship[1] + '' +
151 | ' ' + box +
152 | '
';
153 | });
154 |
155 | return '' + html + '
';
156 | }
157 | };
158 |
159 | /**
160 | * Оповещает пользователя о переходе хода,
161 | * блокирует игровое поле ожидающего
162 | *
163 | * @param fKey строковый ключ игрока
164 | * @returns {*}
165 | */
166 | GameUI.prototype.showProgress = function (fKey) {
167 | // в конце игры, блокируем игровые поля и убираем метку активности с имени
168 | if (!fKey) {
169 | this.fieldHtml.querySelectorAll('table.field').forEach(function (table) {
170 | table.classList.add('timeout');
171 | });
172 |
173 | if (options.printName) {
174 | this.nameHtml.querySelectorAll('.name').forEach(function (span) {
175 | span.classList.remove('act');
176 | });
177 | }
178 | return false;
179 | }
180 |
181 | // делаем игровое поле неактивным
182 | this.fieldHtml.querySelectorAll('table.field.timeout').forEach(function (table) {
183 | table.classList.remove('timeout');
184 | });
185 | this.fieldHtml.querySelector('table.field#' + fKey).classList.add('timeout');
186 |
187 | if (!options.printName)
188 | return null;
189 |
190 | // ставим метку к кому перешел ход
191 | this.nameHtml.querySelectorAll('.name').forEach(function (span) {
192 | span.classList.remove('act');
193 | });
194 | this.nameHtml.querySelector('.name#' + fKey).classList.add('act');
195 | };
196 |
197 | /**
198 | * Вешает событие ожидания клика по игровому полю
199 | *
200 | * @param fKey - строковый ключ игрока
201 | * @param callback - фукция
202 | */
203 | GameUI.prototype.clickToField = function (fKey, callback) {
204 | this.fieldHtml.querySelector('table.field#' + fKey).onclick = callback;
205 | };
206 |
207 | /**
208 | * Печатает маркер на игровом поле
209 | *
210 | * @param point - точка в формате [x, y]
211 | * @param fKey - строковый ключ игрока
212 | * @param tPoint - тип точки
213 | * @param auto - автоматическая точка
214 | * @returns {null}
215 | */
216 | GameUI.prototype.setMarker = function (point, fKey, tPoint, auto) {
217 | var X = point[0],
218 | Y = point[1],
219 | elClass = '';
220 | auto = auto || false;
221 |
222 | switch (tPoint) {
223 | case (options.tPoint.NUL): // мимо
224 | elClass = auto ? 'auto-null' : 'null';
225 | break;
226 | case (options.tPoint.KIL): // ранил
227 | elClass = 'kill';
228 | break;
229 | case (options.tPoint.KIL + '_death'): // убил
230 | elClass = 'kill';
231 | break;
232 | default:
233 | return null;
234 | }
235 |
236 | if (!auto) this.printLog(point, fKey, tPoint);
237 |
238 | var coll = this.fieldHtml.querySelector('table.field#' + fKey)
239 | .rows[X].cells[Y];
240 | coll.querySelector('.box').className += ' ' + elClass;
241 |
242 | if (!auto) {
243 | var shotBox = document.createElement('div');
244 | shotBox.setAttribute('class', 'shot');
245 | coll.appendChild(shotBox);
246 |
247 | h.animateOpacity(shotBox, 2000);
248 | }
249 | };
250 |
251 | /**
252 | * Ставит точки подсказки по углам от подбитого корабля
253 | *
254 | * @param point - точка в формате [x, y]
255 | * @param fKey - строковый ключ игрока
256 | * @param callback - фукция запускается после установки точки
257 | * @returns {boolean}
258 | */
259 | GameUI.prototype.setHelpMarker = function (point, fKey, callback) {
260 | if (!options.printHelp)
261 | return false;
262 |
263 | var sP = [[-1, -1], [1, -1], [1, 1], [-1, 1]],
264 | X = point[0], Y = point[1];
265 |
266 | for (var i = 0; i < sP.length; i++) {
267 | var nX = X + sP[i][0],
268 | nY = Y + sP[i][1];
269 |
270 | if (nX >= 0 && nX < options.fSize.v && nY >= 0 && nY < options.fSize.h) {
271 | this.setMarker([nX, nY], fKey, options.tPoint.NUL, true);
272 | if (typeof callback == 'function')
273 | callback([nX, nY], fKey);
274 | }
275 | }
276 | };
277 |
278 | /**
279 | * Информирует пользователя об оставшихся кораблях на игровом поле
280 | *
281 | * @param ctnCell - массив точек убитого корабля
282 | * @param fKey - строковый ключ игрока
283 | */
284 | GameUI.prototype.shipInfoMap = function (ctnCell, fKey) {
285 | var count = ctnCell.length;
286 |
287 | var letBox = this.fieldHtml
288 | .querySelector('.list-let#' + fKey)
289 | .querySelector('#cell_' + count);
290 |
291 | var span = letBox.querySelector('span'),
292 | dCtn = parseInt(span.getAttribute('data-ctn')) - 1;
293 |
294 | span.setAttribute('data-ctn', dCtn);
295 | span.innerHTML = 'x' + dCtn;
296 |
297 | var shot = document.createElement('div');
298 | shot.setAttribute('class', 'shot');
299 | span.appendChild(shot);
300 |
301 | h.animateOpacity(shot, 3000);
302 |
303 | // если кораблей такова типа больше нет
304 | if (dCtn == 0) {
305 | letBox.querySelectorAll('div').forEach(function (box) {
306 | box.className += ' kill'; // ставим метку убит
307 | });
308 | }
309 |
310 | // если корабль убит, делаем его крассным
311 | var table = this.fieldHtml.querySelector('.field#' + fKey);
312 | ctnCell.forEach(function (ship) {
313 | var X = ship[0],
314 | Y = ship[1];
315 |
316 | table.rows[X].cells[Y]
317 | .querySelector('.box')
318 | .className += ' death';
319 | });
320 | };
321 |
322 | /**
323 | * Ведет журнал боя,
324 | * так же запускает фильтр журнала и подсветку точек на игром поле при наведении на запись
325 | *
326 | * @param point - точка выстрела в формате [x, y]
327 | * @param fKey - строковый ключ игрока
328 | * @param tPoint - тип попадания
329 | * @returns {boolean}
330 | */
331 | GameUI.prototype.printLog = function (point, fKey, tPoint) {
332 | if (!options.printLog)
333 | return false;
334 |
335 | var self = this;
336 |
337 | var html = '',
338 | X = point[0], Y = point[1];
339 |
340 | var tPoint_class = '',
341 | tPoint_name = '';
342 |
343 | var player = fKey == options.player.kUser
344 | ? h.getPlayerName(options.player.kBrain) : h.getPlayerName(options.player.kUser),
345 | __date = new Date(),
346 | time = __date.toLocaleTimeString(),
347 | marker = '';
348 |
349 | switch (tPoint) {
350 | case (options.tPoint.KIL):
351 | tPoint_class = 'war';
352 | tPoint_name = 'log_kill';
353 | break;
354 | case (options.tPoint.KIL + '_death'):
355 | tPoint_class = 'kil';
356 | tPoint_name = 'log_death';
357 | break;
358 | case (options.tPoint.NUL):
359 | tPoint_class = 'nul';
360 | tPoint_name = 'log_past';
361 | break;
362 | }
363 |
364 | marker += options.fMarker.v == 'char' ? h.getLetter(X) : (X + 1);
365 | marker += "x";
366 | marker += options.fMarker.h == 'char' ? h.getLetter(Y) : (Y + 1);
367 |
368 | html +=
369 | '' + marker + '' +
370 | '' + h.getMessage(tPoint_name) + '' +
371 | '' + player + '' +
372 | '' + time + '';
373 |
374 | var li = document.createElement('li');
375 | li.setAttribute('id', fKey);
376 | li.setAttribute('data-x', X);
377 | li.setAttribute('data-y', Y);
378 |
379 | if (this.actFilter == fKey)
380 | li.setAttribute('class', 'no-active');
381 |
382 | li.innerHTML = html;
383 |
384 | this.logHtml.querySelector('ul').insertBefore(li, this.logHtml.querySelector('ul').firstChild);
385 |
386 | // при наведении на лог, подсветка точки игрового поля
387 | li.onmouseover = function (event) {
388 | if (event.target.nodeName != 'LI')
389 | return false;
390 |
391 | var fKey = event.target.getAttribute('id'), // event.target.dataset.fkey,
392 | X = event.target.getAttribute('data-x'), // event.target.dataset.x,
393 | Y = event.target.getAttribute('data-y'); // event.target.dataset.y;
394 |
395 | var td = self.fieldHtml.querySelector('.field#' + fKey).rows[X].cells[Y],
396 | shot = document.createElement('div');
397 |
398 | shot.setAttribute('class', 'shot');
399 | td.appendChild(shot);
400 |
401 | li.onmouseout = function () {
402 | h.animateOpacity(shot, 1000);
403 | };
404 | };
405 |
406 | // фильт журнала
407 | if (this.logFilter == null) {
408 | this.logFilter = this.logHtml.querySelector('.bf-filter');
409 | this.logFilter.classList = 'bf-filter';
410 |
411 | var activeFilter = this.logFilter.querySelector('#full');
412 | this.logFilter.querySelectorAll('span').forEach(function (span) {
413 | span.onclick = function (event) {
414 | if (activeFilter == event.target)
415 | return false;
416 |
417 | if (activeFilter)
418 | activeFilter.classList.remove('active');
419 |
420 | self.actFilter = event.target.getAttribute('id');
421 |
422 | activeFilter = event.target;
423 | activeFilter.classList.add('active');
424 |
425 | self.logHtml.querySelectorAll('li').forEach(function (li) {
426 | li.classList = self.actFilter == li.getAttribute('id') ? 'no-active' : '';
427 | });
428 | }
429 | })
430 | }
431 | };
432 |
433 | /**
434 | * Показывает корабли на игровом поле
435 | *
436 | * @param fKey - строковый ключ игрока
437 | */
438 | GameUI.prototype.showShip = function (fKey) {
439 | var field = this.fields[fKey],
440 | table = this.fieldHtml.querySelector('table#' + fKey);
441 |
442 | for (var X = 0; X < field.length; X++) {
443 | for (var Y = 0; Y < field[X].length; Y++)
444 | if (field[X][Y] == options.tPoint.BAR) {
445 | table
446 | .rows[X].cells[Y]
447 | .querySelector('.box')
448 | .className += ' let';
449 | }
450 | }
451 | };
452 |
453 | /**
454 | * Оповещает пользователя об окончании игры,
455 | * выводит модальное окно
456 | *
457 | * @param winner - строковый ключ игрока, победителя
458 | */
459 | GameUI.prototype.gameOver = function (winner) {
460 | var self = this,
461 | kWinner = options.player.kUser == winner ? 'kUser' : 'kBrain';
462 | var title = '',
463 | elClass = '',
464 | message = '';
465 |
466 | // счетчик побед...
467 | options.winner[kWinner]++;
468 |
469 | if (options.player.kUser == winner) {
470 | elClass = 'winner';
471 | title = h.getMessage('you_winner');
472 | message = h.getMessage('info_winner');
473 | } else {
474 | this.showShip(winner);
475 |
476 | elClass = 'lose';
477 | title = h.getMessage('you_lose');
478 | message = h.getMessage('info_loser');
479 | }
480 |
481 | var contentHtml =
482 | '' +
483 | '' +
484 | '
' + title + '
' +
485 | ' ' + message + '
' +
486 | '';
487 |
488 | // выводит модальное окно с опциями
489 | h.modalWindow(h.getMessage('game_over'), contentHtml, [{
490 | elValue: h.getMessage('new_game'),
491 | onClick: function (modal) {
492 | modal.close();
493 | self.battlefield.run();
494 | }
495 | }, {
496 | elValue: h.getMessage('update_options'),
497 | elClass: 'btn-warning',
498 | onClick: function (modal) {
499 | modal.close();
500 | self.showProgress(false);
501 | self.battlefield.updateConfig();
502 | }
503 | }, {
504 | elValue: h.getMessage('close'),
505 | elClass: 'btn-danger',
506 | onClick: function (modal) {
507 | modal.close();
508 | self.showProgress(false);
509 | }
510 | }]);
511 | };
--------------------------------------------------------------------------------
/dist/battlefield.min.js:
--------------------------------------------------------------------------------
1 | /**
2 | * battlefield-js v1.0.1
3 | * This is classic game Battlefield for browsers, implemented on language JavaScript.
4 | *
5 | * repository: git+https://github.com/alx2das/battlefield-js.git
6 | * bugs: git+https://github.com/alx2das/battlefield-js.git
7 | * homepage: https://alx2das.github.io/battlefield-js/examples/
8 | *
9 | * Copyright 2017, Alexandr Builov
10 | * Date: Sun Feb 05 2017
11 | */
12 | !function(e){"use strict";function t(e,r){if(o instanceof t)return o;if(o=this,"object"==typeof r){for(var i in l)n[i]=r.hasOwnProperty(i)?r[i]:l[i];n=Object.assign(n,s)}else n=Object.assign(l,s);n.htmlSelector=document.querySelector(e)}function r(){if(n.fSize.h<10||n.fSize.h>25||n.fSize.v<10||n.fSize.v>25)throw new RangeError(f.getMessage("err_size_field"));if(!(n.fBarrier instanceof Array)||n.fBarrier.length<1)throw new RangeError(f.getMessage("err_barrier"));this.fields=[],this.fName=[n.player.kUser,n.player.kBrain];for(var e in n.player){for(var t=n.player[e],r=new Array(n.fSize.v),i=0;i '+f.getMessage("new_game")+' '+f.getMessage("options")+"";if(n.htmlSelector.innerHTML="",n.printName){var a=n.winner.kUser+n.winner.kBrain!=0;i+=''+f.getPlayerName(n.player.kUser)+""+(a?''+n.winner.kUser+"":"")+"&"+(a?''+n.winner.kBrain+"":"")+' '+f.getPlayerName(n.player.kBrain)+"
"}if(this.nameHtml=document.createElement("div"),this.nameHtml.setAttribute("class","bf-player-name"),this.nameHtml.innerHTML=i,n.htmlSelector.appendChild(this.nameHtml),this.fieldHtml=document.createElement("div"),this.fieldHtml.setAttribute("class","bf-fields"),this.fieldHtml.innerHTML=this.getFieldHTML(n.player),n.htmlSelector.appendChild(this.fieldHtml),n.printLog){var l=' '+f.getMessage("filter")+': '+f.getPlayerName(n.player.kUser)+' '+f.getPlayerName(n.player.kBrain)+' '+f.getMessage("full")+"
";this.logFilter=null,this.actFilter="",this.logHtml=document.createElement("div"),this.logHtml.setAttribute("class","bf-logger"),this.logHtml.innerHTML=""+f.getMessage("info_log_title")+l+"
",n.htmlSelector.appendChild(this.logHtml)}document.getElementById("new_game").onclick=function(){r.battlefield.run()},document.getElementById("config").onclick=function(){r.battlefield.updateConfig()}}var n={},l={printName:!0,printLog:!0,printHelp:!0,fSize:{h:15,v:15},fBarrier:[[5,2],[4,3],[3,4],[2,5],[1,6]],fMarker:{h:"number",v:"char"}},s={player:{kUser:"FUser",kBrain:"FBrain"},winner:{kUser:0,kBrain:0},tPoint:{DEF:0,BAR:1,KIL:2,NUL:3}},o=!1,c="middle";t.prototype.run=function(){try{var e=new r,t=e.setBarrier();new i(t,this)}catch(e){f.showExceptions(e)}},t.prototype.getLevel=function(e){var t={easy:{fSize:{h:10,v:10},fBarrier:[[4,1],[3,2],[2,3],[1,4]]},middle:{fSize:{h:15,v:15},fBarrier:[[5,2],[4,3],[3,4],[2,5],[1,6]]},hard:{fSize:{h:15,v:20},fBarrier:[[6,1],[5,2],[4,3],[3,4],[2,5],[1,6]]}};if("object"==typeof t[e])return t[e];throw new Error(f.getMessage("err_invalid_level"))},t.prototype.setLevel=function(e){var t=this.getLevel(e);return n.fSize=t.fSize,n.fBarrier=t.fBarrier,c=e,this},t.prototype.updateConfig=function(){function e(e){s.querySelector("#opt-fsize").querySelectorAll("label").forEach(function(t){var r=t.querySelector("input");e?(t.classList="",r.disabled=!1):(t.classList="no-act",r.disabled=!0)}),s.querySelector("#opt-fbarrier").querySelectorAll("label").forEach(function(t){var r=t.querySelector("input");e?(t.classList="",r.disabled=!1):(t.classList="no-act",r.disabled=!0)})}function t(){var e={fSize:{h:0,v:0},fBarrier:[]};return e.fSize.h=parseInt(s.querySelector("input#type-h").value),e.fSize.v=parseInt(s.querySelector("input#type-v").value),s.querySelector("#opt-fbarrier").querySelectorAll("label").forEach(function(t){var r=t.querySelector("input"),i=parseInt(r.getAttribute("bar")),a=parseInt(r.value);a>0&&e.fBarrier.push([i,a])}),e}function r(e){s.querySelector("input#type-h").value=e.fSize.h,s.querySelector("input#type-v").value=e.fSize.v,s.querySelector("#opt-fbarrier").querySelectorAll("input").forEach(function(e){e.value=0}),e.fBarrier.forEach(function(e){var t=e[0],r=e[1];s.querySelector("#opt-fbarrier").querySelector("input#bar_"+t).value=r})}var i=this,a='';a="";var l=c;f.modalWindow(f.getMessage("options"),a,[{elValue:f.getMessage("new_game"),onClick:function(e){e.close();try{if("user"==l){var r=t();n.fSize=r.fSize,n.fBarrier=r.fBarrier,c=l}else l.length>0&&l!=c&&i.setLevel(l);i.run()}catch(e){f.showExceptions(e)}}},{elValue:f.getMessage("set_default_params"),elClass:"btn-warning",onClick:function(e){e.close(),i.setLevel("middle"),i.run()}},{elValue:f.getMessage("close"),elClass:"btn-danger",onClick:function(e){e.close()}}]);var s=document.querySelector(".bf-config"),o=document.getElementById("config_level").querySelectorAll("button");e("user"==c),r(n),o.forEach(function(t){t.onclick=function(t){return l=t.target.value,o.forEach(function(e){e.classList.remove("active")}),t.target.classList+=" active","user"==l?e(!0):(e(!1),r(i.getLevel(l))),!1}})},r.prototype.setBarrier=function(){function e(r,a){if(!(i>0))throw new RangeError(f.getMessage("err_max_iteration"));i--;for(var l=[],s=f.rand(0,n.fSize.v-1),o=f.rand(0,n.fSize.h-1),c=f.rand(1,2)%2,u=0;us||t<0||t>o)return!1;if(r[e][t]!==n.tPoint.DEF)return!1;for(var c=0;c=0&&a<=s&&l>=0&&l<=o&&r[a][l]!=n.tPoint.DEF)return!1;return!0}var r=this,i=100*this.fName.length;return this.fName.forEach(function(t){var i=r.fields[t];n.fBarrier.forEach(function(t){for(var r=t[0],a=t[1],n=0;n0)l=r();else{var o=f.rand(0,i._lsShot[e].length-1);l=i._lsShot[e][o],i._lsShot[e].splice(o,1)}if(l instanceof Array&&"undefined"!==l[0]&&"undefined"!==l[1]){var c=l[0],u=l[1];if(c>=0&&c=0&&u=0&&h=0&&pc?c:m,y=y=0&&p=0&&v';for(var a=0;a";for(var l=0;l'+c+"":"",o=0===l?''+u+"
":""}var d="";t&&(d=i[e][a][l]==n.tPoint.BAR?" let":""),r+=""+s+o+' | '}r+=""}return r+=""}function r(e){var t="";return n.fBarrier.forEach(function(e){for(var r="",i=0;i';t+=' x'+e[1]+" "+r+"
"}),''+t+"
"}var i=this.fields,a="";for(var l in e){var s=e[l],o=s==n.player.kUser?"lf":"rg",c="lf"==o;a+='',a+=t(s,c),a+=r(s),a+="
"}return a},a.prototype.showProgress=function(e){return e?(this.fieldHtml.querySelectorAll("table.field.timeout").forEach(function(e){e.classList.remove("timeout")}),this.fieldHtml.querySelector("table.field#"+e).classList.add("timeout"),n.printName?(this.nameHtml.querySelectorAll(".name").forEach(function(e){e.classList.remove("act")}),void this.nameHtml.querySelector(".name#"+e).classList.add("act")):null):(this.fieldHtml.querySelectorAll("table.field").forEach(function(e){e.classList.add("timeout")}),n.printName&&this.nameHtml.querySelectorAll(".name").forEach(function(e){e.classList.remove("act")}),!1)},a.prototype.clickToField=function(e,t){this.fieldHtml.querySelector("table.field#"+e).onclick=t},a.prototype.setMarker=function(e,t,r,i){var a=e[0],l=e[1],s="";switch(i=i||!1,r){case n.tPoint.NUL:s=i?"auto-null":"null";break;case n.tPoint.KIL:s="kill";break;case n.tPoint.KIL+"_death":s="kill";break;default:return null}i||this.printLog(e,t,r);var o=this.fieldHtml.querySelector("table.field#"+t).rows[a].cells[l];if(o.querySelector(".box").className+=" "+s,!i){var c=document.createElement("div");c.setAttribute("class","shot"),o.appendChild(c),f.animateOpacity(c,2e3)}},a.prototype.setHelpMarker=function(e,t,r){if(!n.printHelp)return!1;for(var i=[[-1,-1],[1,-1],[1,1],[-1,1]],a=e[0],l=e[1],s=0;s=0&&o=0&&c'+p+''+f.getMessage(c)+''+u+''+h+"";var v=document.createElement("li");if(v.setAttribute("id",t),v.setAttribute("data-x",l),v.setAttribute("data-y",s),this.actFilter==t&&v.setAttribute("class","no-active"),v.innerHTML=a,this.logHtml.querySelector("ul").insertBefore(v,this.logHtml.querySelector("ul").firstChild),v.onmouseover=function(e){if("LI"!=e.target.nodeName)return!1;var t=e.target.getAttribute("id"),r=e.target.getAttribute("data-x"),a=e.target.getAttribute("data-y"),n=i.fieldHtml.querySelector(".field#"+t).rows[r].cells[a],l=document.createElement("div");l.setAttribute("class","shot"),n.appendChild(l),v.onmouseout=function(){f.animateOpacity(l,1e3)}},null==this.logFilter){this.logFilter=this.logHtml.querySelector(".bf-filter"),this.logFilter.classList="bf-filter";var m=this.logFilter.querySelector("#full");this.logFilter.querySelectorAll("span").forEach(function(e){e.onclick=function(e){return m!=e.target&&(m&&m.classList.remove("active"),i.actFilter=e.target.getAttribute("id"),m=e.target,m.classList.add("active"),void i.logHtml.querySelectorAll("li").forEach(function(e){e.classList=i.actFilter==e.getAttribute("id")?"no-active":""}))}})}},a.prototype.showShip=function(e){for(var t=this.fields[e],r=this.fieldHtml.querySelector("table#"+e),i=0;i
'+i+"
"+l+"
";f.modalWindow(f.getMessage("game_over"),s,[{elValue:f.getMessage("new_game"),onClick:function(e){e.close(),t.battlefield.run()}},{elValue:f.getMessage("update_options"),elClass:"btn-warning",onClick:function(e){e.close(),t.showProgress(!1),t.battlefield.updateConfig()}},{elValue:f.getMessage("close"),elClass:"btn-danger",onClick:function(e){e.close(),t.showProgress(!1)}}])};var f={playerName:{def:{kUser:"Игрок",kBrain:"Компьютер"}},rand:function(e,t){return e=e||0,t=t||100,parseInt(Math.random()*(t-e+1)+e)},getLetter:function(e,t){var t=t||"",r="ABCDEFGHIJKLMNOPQRSTUVWXYZ";return e>r.length?this.getLetter(e-r.length,""==t?1:t+1):r[e]+t},shuffle:function(e){for(var t,r,i=0;iпопробуешь еще?",info_loser:"В следующий раз повезет больше,
попробуешь еще?"};return"string"==typeof t[e]&&t[e].length>0?t[e]:e},getPlayerName:function(e){var t=e==n.player.kUser?"kUser":"kBrain";return this.playerName.def[t]},showExceptions:function(e){var r="",i="",a=[];switch(e.name){case"Error":r=f.getMessage("info_title_error"),i=e.message,a=[{elValue:f.getMessage("set_default_params"),onClick:function(e){e.close();var r=new t(n.htmlSelector,n);r.setLevel("middle").run()}}];break;case"RangeError":r=f.getMessage("info_title_range_error"),i=e.message,a=[{elValue:f.getMessage("update_options"),onClick:function(e){e.close();var r=new t(n.htmlSelector,n);r.updateConfig()}},{elValue:f.getMessage("set_default_params"),elClass:"btn-warning",onClick:function(e){e.close();var r=new t(n.htmlSelector,n);r.setLevel("middle").run()}}];break;default:r=e.name,i=e.message}this.modalWindow(r,i,a)},modalWindow:function(e,t,r){function i(e,t){return function(r){"function"==typeof t[e].onClick&&t[e].onClick(a,r)}}var a={font:!1,window:!1,construct:function(){if(this.createHTML().autoPosition(),!(r instanceof Array)||0==r.length)return this.window.querySelector(".footer").style.display="none",!1;for(var e=this.window.querySelector(".footer"),t=[],a=0;a '+t+'
',document.body.appendChild(this.font),document.body.appendChild(this.window),this},autoPosition:function(){var e=this.window.clientWidth/2,t=this.window.clientHeight/2;this.window.style.marginTop=t*-1+"px",this.window.style.marginLeft=e*-1+"px"},close:function(){document.body.removeChild(this.font),document.body.removeChild(this.window)}};a.construct()},animateOpacity:function(e,t){e.style.opacity=1;var r=setInterval(function(){if(e.style.opacity=e.style.opacity-100/(t/.1),e.style.opacity<=0){clearInterval(r);try{e.parentNode.removeChild(e)}catch(e){}}},1)}};e.Battlefield=t}(window),function(e,t,r){(t[r]=t[r]||[]).push(function(){try{t.yaCounter42568409=new Ya.Metrika({id:42568409,clickmap:!0,trackLinks:!0,accurateTrackBounce:!0})}catch(e){}});var i=e.getElementsByTagName("script")[0],a=e.createElement("script"),n=function(){i.parentNode.insertBefore(a,i)};a.type="text/javascript",a.async=!0,a.src="https://mc.yandex.ru/metrika/watch.js","[object Opera]"==t.opera?e.addEventListener("DOMContentLoaded",n,!1):n()}(document,window,"yandex_metrika_callbacks");
--------------------------------------------------------------------------------
/src/css/battlefield.helper.css:
--------------------------------------------------------------------------------
1 | .bf-go-smile {
2 | float: left;
3 | width: 100px;
4 | height: 100px;
5 | margin-left: 90px;
6 | }
7 |
8 | .bf-go-smile.winner {
9 | background-image: url();
10 | }
11 |
12 | .bf-go-smile.lose {
13 | background-image: url();
14 | }
15 |
16 | .bf-go-text {
17 | float: left;
18 | margin: 0 0 0 35px;
19 | padding: 15px 0;
20 | }
21 |
22 | .bf-go-text h2 {
23 | margin-bottom: 10px
24 | }
25 |
26 | table.bf-config {
27 | width: 100%;
28 | border-collapse: collapse;
29 | }
30 |
31 | table.bf-config td {
32 | padding: 10px 0
33 | }
34 |
35 | table.bf-config td.name {
36 | width: 28%;
37 | font-size: 13px;
38 | font-weight: 700;
39 | vertical-align: top
40 | }
41 |
42 | table.bf-config td.name.top {
43 | padding: 15px 0
44 | }
45 |
46 | table.bf-config td.options {
47 | width: 73%
48 | }
49 |
50 | table.bf-config td .border {
51 | border-top: 1px solid rgba(34, 36, 38, .15)
52 | }
53 |
54 | table.bf-config td.options .btn {
55 | padding: 4px 14px;
56 | border-radius: 20px;
57 | background: #fff;
58 | border: 1px solid rgba(34, 36, 38, .15);
59 | color: rgba(0, 0, 0, .8);
60 | }
61 |
62 | table.bf-config td.options .btn:hover {
63 | background-color: #F9FAFB;
64 | }
65 |
66 | table.bf-config td.options .btn.active {
67 | background-color: #e7ff7f !important;
68 | }
69 |
70 | table.bf-config td.options .btn.rg {
71 | float: right;
72 | margin: 0;
73 | color: #794B02;
74 | background-color: #ffeacc;
75 | }
76 |
77 | table.bf-config td.options label {
78 | display: inline-block;
79 | margin: 2px 25px 2px 0;
80 | cursor: pointer
81 | }
82 |
83 | table.bf-config td.options label input[type=number] {
84 | color: rgba(0, 0, 0, .8);
85 | width: 45px;
86 | font-size: 13px;
87 | border: 1px solid rgba(34, 36, 38, .15);
88 | margin: -3px 0 0 0;
89 | padding: 3px 4px;
90 | border-radius: 3px;
91 | background-color: rgba(255, 249, 0, 0.25);
92 | }
93 |
94 | table.bf-config td.options label.no-act input[type=number] {
95 | border-color: #fff;
96 | background: none
97 | }
98 |
99 | table.bf-config td.options input[type=checkbox], table.bf-config td.options input[type=radio] {
100 | float: left;
101 | margin-top: 2px;
102 | width: auto;
103 | margin-right: 5px;
104 | }
105 |
106 | table.bf-config td.options.no-label label {
107 | cursor: default !important;
108 | }
109 |
110 | table.bf-config td.options div {
111 | padding-bottom: 4px
112 | }
113 |
114 | table.bf-config td.options div.top {
115 | padding-top: 10px
116 | }
--------------------------------------------------------------------------------
/dist/battlefield.min.css:
--------------------------------------------------------------------------------
1 | /**
2 | * battlefield-js v1.0.1
3 | * This is classic game Battlefield for browsers, implemented on language JavaScript.
4 | *
5 | * repository: git+https://github.com/alx2das/battlefield-js.git
6 | * bugs: git+https://github.com/alx2das/battlefield-js.git
7 | * homepage: https://alx2das.github.io/battlefield-js/examples/
8 | *
9 | * Copyright 2017, Alexandr Builov
10 | * Date: Sun Feb 05 2017
11 | */
12 | .container,body,html{min-height:100%}.bf-fields{background-color:#7a8b9f}.bf-fields .board{width:50%}.bf-fields .board.lf{float:left}.bf-fields .board.rg{float:right}.bf-fields table.field{border-collapse:collapse;margin:25px}.bf-fields table.field td,.bf-fields table.field td .box{border:2px solid #E5EAFF}.bf-fields table.field td .box,.bf-fields table.field td .ll,.bf-fields table.field td .mm{width:28px;height:28px;line-height:28px;text-align:center}.bf-fields table.field td .ll,.bf-fields table.field td .mm{position:absolute;z-index:1;font-family:Georgia;color:#afafaf;font-weight:700;font-size:12px}.bf-fields table.field td .ll{left:-28px}.bf-fields table.field td .mm{top:-28px}.bf-fields table.field td{position:relative}.bf-fields table.field td .box{float:left;margin:-2px}.bf-fields .list-let .box.kill,.bf-fields table.field td .box.auto-null,.bf-fields table.field td .box.kill,.bf-fields table.field td .box.null{background:url() no-repeat}.bf-fields table.field td .box.let{background-color:#E5EAFF}.bf-fields table.field td .box.null{opacity:1!important}.bf-fields table.field td .box.auto-null{opacity:.5}.bf-fields table.field td .box.kill{background-position:-28px 0;background-color:#a1b9d6}.bf-fields table.field td .box.death{background-color:#deabab}.bf-fields .shot{position:absolute;z-index:1;width:28px;height:28px;margin:-2px;background-color:rgba(255,249,0,.7);border:2px solid #fff900}.bf-fields .list-let{margin:25px 15px 0;float:left}.bf-fields .list-let .let{margin-bottom:10px}.bf-fields .list-let .let:after{content:" ";display:block;clear:both}.bf-fields .list-let span{width:26px;height:26px;line-height:26px;margin:0 10px;text-align:center;color:#afafaf;font-weight:700;display:inline-block;background:0 0;border:none}.bf-fields:after,.bf-logger:after,.bf-player-name:after,.ctn-footer:after,.ctn-header:after{display:block;content:" ";clear:both}.bf-fields .list-let span .shot{width:24px;height:24px;margin:-27px 0 0}.bf-fields .list-let .box{float:left;width:24px;height:24px;margin-left:-2px;background-color:rgba(165,188,208,.5)!important;border:2px solid #E5EAFF}.ctn-footer .file a:hover,.ctn-header{border-bottom:1px solid #7a8b9f}.bf-fields .list-let .box.kill{background-position:-28px 0}.bf-fields .board.rg table.field td .box{cursor:pointer}.bf-fields .board.rg table.field td .box:hover{box-shadow:inset 0 0 5px 2px #ff8d8d}.bf-fields .board.rg table.field td .box.auto-null:hover,.bf-fields .board.rg table.field td .box.kill:hover,.bf-fields .board.rg table.field td .box.let:hover,.bf-fields .board.rg table.field td .box.null:hover{box-shadow:none!important;cursor:default!important}.bf-fields .board table.field.timeout{opacity:.7;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.bf-fields .board table.field.timeout .box{cursor:wait!important}.bf-fields .board.lf .list-let,.bf-fields .board.lf .list-let .box,.bf-fields .board.lf table.field,.bf-fields .board.rg .list-let span{float:right}.bf-fields .board.lf .list-let span,.bf-fields .board.rg .list-let,.bf-fields .board.rg .list-let .box,.bf-fields .board.rg table.field{float:left}*{margin:0;padding:0}body,html{height:100%;font-family:Arial,sans-serif}.wrap{width:950px;margin:0 auto}.container .clear-footer{height:46px}.ctn-header{padding:20px 0}.ctn-header .logotype{float:left;color:#7a8b9f;font-weight:700;font-size:26px;text-decoration:none}.ctn-header .logotype small{font-weight:400;font-size:12px;color:#b5b5b5}.ctn-header .soc-btn{float:right;padding:3px 0}.btn,.ctn-footer .copy{float:left}.ctn-footer{padding:15px 0;margin-top:-46px;border-top:1px solid #7a8b9f;font-size:13px;color:#b5b5b5}.ctn-footer .file{float:left;margin-left:25px}.ctn-footer .file a{color:#7a8b9f;text-decoration:none}.bf-fields,.bf-logger,.bf-modal-window,.bf-player-name,.btn{font-family:Arial,sans-serif;color:#7a8b9f;font-size:13px}.bf-fields,.bf-logger,.bf-player-name{padding:25px 0}.btn{margin-right:10px;outline:0;margin-bottom:0;font-weight:400;text-align:center;vertical-align:middle;cursor:pointer;border:1px solid transparent;white-space:nowrap;padding:7px 14px;font-size:13px;line-height:1.2;border-radius:3px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:#5cb85c;color:#fff}.btn:active,.btn:hover{background-color:#449d44}.btn.btn-warning{background-color:#f0ad4e}.btn.btn-warning:active,.btn.btn-warning:hover{background-color:#ec971f}.btn.btn-danger{background-color:#d9534f;float:right;margin-right:0}.btn.btn-danger:active,.btn.btn-danger:hover{background-color:#c9302c}.btn.btn-danger:active{background-image:none}span.js{margin-left:15px;cursor:pointer;color:#7a8b9f;border-bottom:1px dashed #7a8b9f}span.js:hover{border-bottom:none}.bf-go-smile{float:left;width:100px;height:100px;margin-left:90px}.bf-go-smile.winner{background-image:url()}.bf-go-smile.lose{background-image:url()}.bf-go-text{float:left;margin:0 0 0 35px;padding:15px 0}.bf-go-text h2{margin-bottom:10px}table.bf-config{width:100%;border-collapse:collapse}table.bf-config td{padding:10px 0}table.bf-config td.name{width:28%;font-size:13px;font-weight:700;vertical-align:top}table.bf-config td.name.top{padding:15px 0}table.bf-config td.options{width:73%}table.bf-config td .border{border-top:1px solid rgba(34,36,38,.15)}table.bf-config td.options .btn{padding:4px 14px;border-radius:20px;background:#fff;border:1px solid rgba(34,36,38,.15);color:rgba(0,0,0,.8)}table.bf-config td.options .btn:hover{background-color:#F9FAFB}table.bf-config td.options .btn.active{background-color:#e7ff7f!important}table.bf-config td.options .btn.rg{float:right;margin:0;color:#794B02;background-color:#ffeacc}table.bf-config td.options label{display:inline-block;margin:2px 25px 2px 0;cursor:pointer}table.bf-config td.options label input[type=number]{color:rgba(0,0,0,.8);width:45px;font-size:13px;border:1px solid rgba(34,36,38,.15);margin:-3px 0 0;padding:3px 4px;border-radius:3px;background-color:rgba(255,249,0,.25)}table.bf-config td.options label.no-act input[type=number]{border-color:#fff;background:0 0}table.bf-config td.options input[type=checkbox],table.bf-config td.options input[type=radio]{float:left;margin-top:2px;width:auto;margin-right:5px}table.bf-config td.options.no-label label{cursor:default!important}table.bf-config td.options div{padding-bottom:4px}table.bf-config td.options div.top{padding-top:10px}.bf-logger{max-width:950px;margin:0 auto}.bf-logger h2{padding-bottom:15px}.bf-logger ul{list-style:none;max-height:300px;overflow-y:auto;margin-bottom:25px}.bf-logger ul li{display:block;padding:7px 0;border-bottom:1px solid #d6d6d6}.bf-logger ul li.no-active{display:none}.bf-logger ul li:hover{background-color:#f6f6f6}.game-log ul li:last-child{border-bottom:none}.bf-logger ul li .point,.bf-logger ul li .time,.bf-logger ul li .type{width:60px;display:inline-block}.bf-logger ul li .point,.bf-logger ul li .time{font-weight:700}.bf-logger ul li .type{margin:0 10px;font-size:12px;padding:2px 0;text-align:center;border-radius:2px;font-weight:700}.bf-logger ul li .type.kil{background-color:#d9534f;color:#fff}.bf-logger ul li .type.war{background-color:#f0ad4e;color:#fff}.bf-logger ul li .name{font-size:14px}.bf-logger ul li .time{float:right}.bf-logger .bf-filter{float:right;font-weight:400;font-size:13px}.bf-logger .bf-filter.no-active{display:none}.bf-logger .bf-filter span.active{border-bottom:1px solid}.bf-modal-font{position:fixed;z-index:9;top:0;left:0;width:100%;height:100%;background:rgba(122,139,159,.8)}.bf-modal-window{position:fixed;z-index:19;top:50%;left:50%;width:600px;background:#fff;border-radius:4px}.bf-modal-window .content{padding:25px 15px 15px}.bf-modal-window .content span{float:right;font-size:12px;font-weight:400;padding:2px 0;color:#7A8BB4;font-style:italic}.bf-modal-window .header{font-weight:700;font-size:16px;padding-bottom:16px;border-bottom:2px solid rgba(122,139,159,.3)}.bf-modal-window .cont{padding:30px 0}.bf-modal-window .footer{padding-top:15px;border-top:2px solid rgba(122,139,159,.3)}.bf-modal-window .cont:after,.bf-modal-window .footer:after,.bf-modal-window .header:after{content:" ";clear:both;display:block}.bf-player-name{text-align:center;width:950px;margin:0 auto}.bf-player-name span{display:inline-block;padding:0 10px;font-size:15px;height:24px;line-height:24px}.bf-player-name .name{width:120px;background:#7a8b9f;color:#fff;margin:0 10px 0 6px;padding:4px 10px;border-radius:3px;height:auto;line-height:1.1}.bf-player-name .name.act{background:#f0ad4e}.bf-player-name .total{min-width:30px;text-align:center;color:#7a8b9f;font-weight:700}.bf-player-name .opt{float:left}.bf-player-name .left{float:left;margin-top:-5px}.bf-player-name .center{width:460px;margin:0 auto}.bf-player-name .center:after{content:" ";clear:both;display:block}.bf-player-name .right{float:right}.bf-player-name .left a{color:#7a8b9f;font-weight:700;font-size:26px;text-decoration:none}.bf-player-name .right span{padding:0}
--------------------------------------------------------------------------------