From 3a0cf3f14d637b8cebc527d3825f6ea6d12eed7a Mon Sep 17 00:00:00 2001 From: zhongjin Date: Wed, 16 Jan 2019 11:59:25 +0800 Subject: [PATCH] Initial commit --- .gitignore | 9 + .npmignore | 12 + .travis.yml | 24 + LICENSE | 21 + README.md | 304 +++ admin/i18n/de/translations.json | 74 + admin/i18n/en/translations.json | 74 + admin/i18n/es/translations.json | 74 + admin/i18n/fr/translations.json | 74 + admin/i18n/it/translations.json | 74 + admin/i18n/nl/translations.json | 74 + admin/i18n/pl/translations.json | 74 + admin/i18n/pt/translations.json | 74 + admin/i18n/ru/translations.json | 74 + admin/img/plc_back.png | Bin 0 -> 68765 bytes admin/index.html | 1372 ++++++++++++ admin/lib/css/jsgrid-theme.css | 205 ++ admin/lib/css/jsgrid-theme.min.css | 7 + admin/lib/css/jsgrid.css | 121 + admin/lib/css/jsgrid.min.css | 7 + admin/lib/js/grid.locale-de.js | 10 + admin/lib/js/grid.locale-ru.js | 11 + admin/lib/js/jsgrid.js | 1954 +++++++++++++++++ admin/modbus.png | Bin 0 -> 5025 bytes admin/words.js | 83 + appveyor.yml | 25 + gulpfile.js | 401 ++++ img/img1.png | Bin 0 -> 10952 bytes img/img2.png | Bin 0 -> 12875 bytes img/img3.png | Bin 0 -> 13335 bytes img/img4.png | Bin 0 -> 11851 bytes img/img5.png | Bin 0 -> 60305 bytes io-package.json | 84 + lib/common.js | 335 +++ lib/jsmodbus/README.md | 206 ++ lib/jsmodbus/handler/client/ReadCoils.js | 56 + .../handler/client/ReadDiscreteInputs.js | 62 + .../handler/client/ReadHoldingRegisters.js | 52 + .../handler/client/ReadInputRegisters.js | 52 + .../handler/client/WriteMultipleCoils.js | 73 + .../handler/client/WriteMultipleRegisters.js | 71 + .../handler/client/WriteSingleCoil.js | 47 + .../handler/client/WriteSingleRegister.js | 50 + lib/jsmodbus/handler/server/ReadCoils.js | 52 + .../handler/server/ReadDiscreteInputs.js | 69 + .../handler/server/ReadHoldingRegisters.js | 55 + .../handler/server/ReadInputRegisters.js | 52 + .../handler/server/WriteMultipleCoils.js | 77 + .../handler/server/WriteMultipleRegisters.js | 60 + .../handler/server/WriteSingleCoil.js | 62 + .../handler/server/WriteSingleRegister.js | 50 + lib/jsmodbus/index.js | 17 + lib/jsmodbus/modbus-client-core.js | 145 ++ lib/jsmodbus/modbus-server-core.js | 91 + lib/jsmodbus/package.json | 52 + .../transports/modbus-client-serial.js | 148 ++ .../transports/modbus-client-tcp-rtu.js | 175 ++ lib/jsmodbus/transports/modbus-client-tcp.js | 156 ++ lib/jsmodbus/transports/modbus-server-tcp.js | 149 ++ lib/master.js | 650 ++++++ lib/slave.js | 406 ++++ lib/utils.js | 83 + main.js | 1211 ++++++++++ package.json | 42 + test/Ananas32.zip | Bin 0 -> 1053450 bytes test/Ananas64.zip | Bin 0 -> 1312008 bytes test/RMMS.zip | Bin 0 -> 563348 bytes test/lib/setup.js | 728 ++++++ test/mod_RSsim.exe | Bin 0 -> 1205248 bytes test/test.js | 20 + test/testAdapter.js | 141 ++ test/testPackageFiles.js | 91 + 72 files changed, 11102 insertions(+) create mode 100644 .gitignore create mode 100644 .npmignore create mode 100644 .travis.yml create mode 100644 LICENSE create mode 100644 README.md create mode 100644 admin/i18n/de/translations.json create mode 100644 admin/i18n/en/translations.json create mode 100644 admin/i18n/es/translations.json create mode 100644 admin/i18n/fr/translations.json create mode 100644 admin/i18n/it/translations.json create mode 100644 admin/i18n/nl/translations.json create mode 100644 admin/i18n/pl/translations.json create mode 100644 admin/i18n/pt/translations.json create mode 100644 admin/i18n/ru/translations.json create mode 100644 admin/img/plc_back.png create mode 100644 admin/index.html create mode 100644 admin/lib/css/jsgrid-theme.css create mode 100644 admin/lib/css/jsgrid-theme.min.css create mode 100644 admin/lib/css/jsgrid.css create mode 100644 admin/lib/css/jsgrid.min.css create mode 100644 admin/lib/js/grid.locale-de.js create mode 100644 admin/lib/js/grid.locale-ru.js create mode 100644 admin/lib/js/jsgrid.js create mode 100644 admin/modbus.png create mode 100644 admin/words.js create mode 100644 appveyor.yml create mode 100644 gulpfile.js create mode 100644 img/img1.png create mode 100644 img/img2.png create mode 100644 img/img3.png create mode 100644 img/img4.png create mode 100644 img/img5.png create mode 100644 io-package.json create mode 100644 lib/common.js create mode 100644 lib/jsmodbus/README.md create mode 100644 lib/jsmodbus/handler/client/ReadCoils.js create mode 100644 lib/jsmodbus/handler/client/ReadDiscreteInputs.js create mode 100644 lib/jsmodbus/handler/client/ReadHoldingRegisters.js create mode 100644 lib/jsmodbus/handler/client/ReadInputRegisters.js create mode 100644 lib/jsmodbus/handler/client/WriteMultipleCoils.js create mode 100644 lib/jsmodbus/handler/client/WriteMultipleRegisters.js create mode 100644 lib/jsmodbus/handler/client/WriteSingleCoil.js create mode 100644 lib/jsmodbus/handler/client/WriteSingleRegister.js create mode 100644 lib/jsmodbus/handler/server/ReadCoils.js create mode 100644 lib/jsmodbus/handler/server/ReadDiscreteInputs.js create mode 100644 lib/jsmodbus/handler/server/ReadHoldingRegisters.js create mode 100644 lib/jsmodbus/handler/server/ReadInputRegisters.js create mode 100644 lib/jsmodbus/handler/server/WriteMultipleCoils.js create mode 100644 lib/jsmodbus/handler/server/WriteMultipleRegisters.js create mode 100644 lib/jsmodbus/handler/server/WriteSingleCoil.js create mode 100644 lib/jsmodbus/handler/server/WriteSingleRegister.js create mode 100644 lib/jsmodbus/index.js create mode 100644 lib/jsmodbus/modbus-client-core.js create mode 100644 lib/jsmodbus/modbus-server-core.js create mode 100644 lib/jsmodbus/package.json create mode 100644 lib/jsmodbus/transports/modbus-client-serial.js create mode 100644 lib/jsmodbus/transports/modbus-client-tcp-rtu.js create mode 100644 lib/jsmodbus/transports/modbus-client-tcp.js create mode 100644 lib/jsmodbus/transports/modbus-server-tcp.js create mode 100644 lib/master.js create mode 100644 lib/slave.js create mode 100644 lib/utils.js create mode 100644 main.js create mode 100644 package.json create mode 100644 test/Ananas32.zip create mode 100644 test/Ananas64.zip create mode 100644 test/RMMS.zip create mode 100644 test/lib/setup.js create mode 100644 test/mod_RSsim.exe create mode 100644 test/test.js create mode 100644 test/testAdapter.js create mode 100644 test/testPackageFiles.js diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..37aa01f --- /dev/null +++ b/.gitignore @@ -0,0 +1,9 @@ +/.idea +/node_modules +.git +admin/i18n/flat.txt +admin/i18n/*/flat.txt +test/Ananas64.exe +iob_npm.done +tmp +package-lock.json \ No newline at end of file diff --git a/.npmignore b/.npmignore new file mode 100644 index 0000000..8f071f6 --- /dev/null +++ b/.npmignore @@ -0,0 +1,12 @@ +gulpfile.js +tasks +node_modules +.idea +.git +/node_modules +test +.travis.yml +appveyor.yml +admin/i18n +iob_npm.done +package-lock.json diff --git a/.travis.yml b/.travis.yml new file mode 100644 index 0000000..c7a2beb --- /dev/null +++ b/.travis.yml @@ -0,0 +1,24 @@ +os: + - linux + - osx +language: node_js +node_js: + - '4' + - '6' + - '8' + - '10' +before_install: + - 'if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then export CC=clang++; export CXX=clang++; export CXXFLAGS=-std=c++11; fi' + - 'if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then export CXX=g++-4.8; fi' +before_script: + - export NPMVERSION=$(echo "$($(which npm) -v)"|cut -c1) + - 'if [[ $NPMVERSION == 5 ]]; then npm install -g npm@5; fi' + - npm -v + - npm install winston@2.3.1 + - 'npm install https://github.com/yunkong2/yunkong2.js-controller/tarball/master --production' +addons: + apt: + sources: + - ubuntu-toolchain-r-test + packages: + - g++-4.8 diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..8f8d28e --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2015-2018 Bluefox + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..2dffc1f --- /dev/null +++ b/README.md @@ -0,0 +1,304 @@ +![Logo](admin/modbus.png) +# yunkong2.rsonoff +===================== + +[![NPM version](http://img.shields.io/npm/v/yunkong2.rsonoff.svg)](https://www.npmjs.com/package/yunkong2.rsonoff) +[![Downloads](https://img.shields.io/npm/dm/yunkong2.rsonoff.svg)](https://www.npmjs.com/package/yunkong2.rsonoff) + +[![NPM](https://nodei.co/npm/yunkong2.rsonoff.png?downloads=true)](https://nodei.co/npm/yunkong2.rsonoff/) + +Implementation of ModBus Slave and Master for yunkong2. Following types are supported: +- Modbus RTU over serial (master) +- Modbus RTU over TCP (master) +- Modbus TCP (slave, master) + +## Settings +### Partner IP Address +IP address of modbus partner. + +### Port +TCP Port of modbus partner if configured as master (client) or own port if configured as slave(server). + +### Device ID +Modbus Device ID. Important if TCP/Modbus bridge is used. + +### Type +Slave(Server) or Master(Client). + +### Use aliases as address +Normally all registers can have address from 0 to 65535. By using of aliases you can define virtual address fields for every type of registers. Normally: +- discrete inputs are from 10001 to 20000 +- coils are from 1 to 1000 +- input registers are from 30001 to 40000 +- holding registers are from 40001 to 60000 + +Every alias will be mapped internally to address, e.g. 30011 will be mapped to input register 10. and so on. + +### Do not align addresses to word +Normally the coils and the discrete inputs addresses are aligned to 16 bit. Like addresses from 3 to 20 will be aligned to 0 up 32. +If this option is active the addresses will not be aligned. + +### Round Real to +How many digits after comma for float and doubles. + +### Poll delay +Cyclic poll interval (Only relevant for master) + +### Reconnect time +Reconnection interval (Only relevant for master) + +### Pulse time +if pulse used for coils, this define the interval how long is pulse. + +### Max read request length +Maximal length of command READ_MULTIPLE_REGISTERS as number of registers to read. + +Some systems require first "write request" to deliver the data on "read request". +You can force this mode by setting of the "Max read request length" to 1. + +**Notice:** +Some USB Modbus solutions (e.g. based on socat) can have trouble to work with serialport npm module. + +There is a software [**Modbus RTU <-> Modbus RTU over TCP**](http://mbus.sourceforge.net/index.html) gateway to enable using of serial RTU over TCP protocol. + +Both solutions **RTU over TCP** and **TCP** works well. + +## Data types + +- uint16be - Unsigned 16 bit (Big Endian): AABB => AABB +- uint16le - Unsigned 16 bit (Little Endian): AABB => BBAA +- int16be - Signed 16 bit (Big Endian): AABB => AABB +- int16le - Signed 16 bit (Little Endian): AABB => BBAA +- uint32be - Unsigned 32 bit (Big Endian): AABBCCDD => AABBCCDD +- uint32le - Unsigned 32 bit (Little Endian): AABBCCDD => DDCCBBAA +- uint32sw - Unsigned 32 bit (Big Endian Word Swap): AABBCCDD => CCDDAABB +- uint32sb - Unsigned 32 bit (Big Endian Byte Swap): AABBCCDD => DDCCBBAA +- int32be - Signed 32 bit (Big Endian): AABBCCDD => AABBCCDD +- int32le - Signed 32 bit (Little Endian): ABBCCDD => DDCCBBAA +- int32sw - Signed 32 bit (Big Endian Word Swap): AABBCCDD => CCDDAABB +- int32sb - Signed 32 bit (Big Endian Byte Swap): AABBCCDD => DDCCBBAA +- uint64be - Unsigned 64 bit (Big Endian): AABBCCDDEEFFGGHH => AABBCCDDEEFFGGHH +- uint64le - Unsigned 64 bit (Little Endian): AABBCCDDEEFFGGHH => HHGGFFEEDDCCBBAA +- uint8be - Unsigned 8 bit (Big Endian): AA => AA +- uint8le - Unsigned 8 bit (Little Endian): AA => AA +- int8be - Signed 8 bit (Big Endian): AA => AA +- int8le - Signed 8 bit (Little Endian): AA => AA +- floatbe - Float (Big Endian): AABBCCDD => AABBCCDD +- floatle - Float (Little Endian): AABBCCDD => DDCCBBAA +- floatsw - Float (Big Endian Word Swap): AABBCCDD => CCDDAABB +- floatsb - Float (Big Endian Byte Swap): AABBCCDD => DDCCBBAA +- doublebe - Double (Big Endian): AABBCCDDEEFFGGHH => AABBCCDDEEFFGGHH +- doublele - Double (Little Endian): AABBCCDDEEFFGGHH => HHGGFFEEDDCCBBAA +- string - String (Zero-end): ABCDEF\0 => ABCDEF\0 +- stringle - String (Little Endian, Zero-end): BADCFE\0 => ABCDEF\0 + +Following description was copied from [here](http://www.chipkin.com/how-real-floating-point-and-32-bit-data-is-encoded-in-modbus-rtu-messages/) + +The point-to-point Modbus protocol is a popular choice for RTU communications if for no other reason that it’s basic convenience. The protocol itself controls the interactions of each device on a Modbus network, how device establishes a known address, how each device recognizes its messages and how basic information is extracted from the data. In essence, the protocol is the foundation of the entire Modbus network. + +Such convenience does not come without some complications however, and Modbus RTU Message protocol is no exception. The protocol itself was designed based on devices with a 16-bit register length. Consequently, special considerations were required when implementing 32-bit data elements. This implementation settled on using two consecutive 16-bit registers to represent 32 bits of data or essentially 4 bytes of data. It is within these 4 bytes of data that single-precision floating point data can be encoded into a Modbus RTU message. + +### The Importance of Byte Order + +Modbus itself does not define a floating point data type but it is widely accepted that it implements 32-bit floating point data using the IEEE-754 standard. However, the IEEE standard has no clear cut definition of byte order of the data payload. Therefore the most important consideration when dealing with 32-bit data is that data is addressed in the proper order. + +For example, the number 123/456.00 as defined in the IEEE 754 standard for single-precision 32-bit floating point numbers appears as follows: + +![Image1](img/img1.png) + +The affects of various byte orderings are significant. For example, ordering the 4 bytes of data that represent 123456.00 in a “B A D C” sequence in known as a “byte swap”. When interpreted as an IEEE 744 floating point data type, the result is quite different: + +![Image2](img/img2.png) + +Ordering the same bytes in a “C D A B” sequence is known as a “word swap”. Again, the results differ drastically from the original value of 123456.00: + +![Image3](img/img3.png) + +Furthermore, both a “byte swap” and a “word swap” would essentially reverse the sequence of the bytes altogether to produce yet another result: + +![Image4](img/img4.png) + +Clearly, when using network protocols such as Modbus, strict attention must be paid to how bytes of memory are ordered when they are transmitted, also known as the ‘byte order’. + +### Determining Byte Order + +The Modbus protocol itself is declared as a ‘big-Endian’ protocol, as per the Modbus Application Protocol Specification, V1.1.b: + +```Modbus uses a “big-Endian” representation for addresses and data items. This means that when a numerical quantity larger than a single byte is transmitted, the most significant byte is sent first.``` + +Big-Endian is the most commonly used format for network protocols – so common, in fact, that it is also referred to as ‘network order’. + +Given that the Modbus RTU message protocol is big-Endian, in order to successfully exchange a 32-bit datatype via a Modbus RTU message, the endianness of both the master and the slave must considered. Many RTU master and slave devices allow specific selection of byte order particularly in the case of software-simulated units. One must merely insure that both all units are set to the same byte order. + +As a rule of thumb, the family of a device’s microprocessor determines its endianness. Typically, the big-Endian style (the high-order byte is stored first, followed by the low-order byte) is generally found in CPUs designed with a Motorola processor. The little-Endian style (the low-order byte is stored first, followed by the high-order byte) is generally found in CPUs using the Intel architecture. It is a matter of personal perspective as to which style is considered ‘backwards’. + +If, however, byte order and endianness is not a configurable option, you will have to determine the how to interpret the byte. This can be done requesting a known floating-point value from the slave. If an impossible value is returned, i.e. a number with a double-digit exponent or such, the byte ordering will most likely need modification. + +### Practical Help + +The FieldServer Modbus RTU drivers offer several function moves that handle 32-bit integers and 32-bit float values. More importantly, these function moves consider all different forms of byte sequencing. The following table shows the FieldServer function moves that copy two adjacent 16-bit registers to a 32-bit integer value. + +| Function Keyword | Swap Mode | Source Bytes | Target Bytes | +|-------------------|--------------------|-----------------|--------------| +| 2.i16-1.i32 | N/A | [ a b ] [ c d ] | [ a b c d ] | +| 2.i16-1.i32-s | byte and word swap | [ a b ] [ c d ] | [ d c b a ] | +| 2.i16-1.i32-sb | byte swap | [ a b ] [ c d ] | [ b a d c ] | +| 2.i16-1.i32-sw | word swap | [ a b ] [ c d ] | [ c d a b ] | + +The following table shows the FieldServer function moves that copy two adjacent 16-bit registers to a 32-bit floating point value: + +| Function Keyword | Swap Mode | Source Bytes | Target Bytes | +|-------------------|--------------------|-----------------|--------------| +| 2.i16-1.ifloat | N/A | [ a b ] [ c d ] | [ a b c d ] | +| 2.i16-1.ifloat-s | byte and word swap | [ a b ] [ c d ] | [ d c b a ] | +| 2.i16-1.ifloat-sb | byte swap | [ a b ] [ c d ] | [ b a d c ] | +| 2.i16-1.ifloat-sw | word swap | [ a b ] [ c d ] | [ c d a b ] | + +The following table shows the FieldServer function moves that copy a single 32-bit floating point value to two adjacent 16-bit registers: + +| Function Keyword | Swap Mode | Source Bytes | Target Bytes | +|------------------|-------------------|-----------------|----------------| +| 1.float-2.i16 |N/A | [ a b ] [ c d ] | [ a b ][ c d ] | +| 1.float-2.i16-s |byte and word swap | [ a b ] [ c d ] | [ d c ][ b a ] | +| 1.float-2.i16-sb |byte swap | [ a b ] [ c d ] | [ b a ][ d c ] | +| 1.float-2.i16-sw |word swap | [ a b ] [ c d ] | [ c d ][ a b ] | + +Given the various FieldServer function moves, the correct handling of 32-bit data is dependent on choosing the proper one. Observe the following behavior of these FieldServer function moves on the known single-precision decimal float value of 123456.00: + +|16-bit Values | Function Move | Result | Function Move | Result | +|---------------|-------------------|-----------|-------------------|---------------| +|0x2000 0x47F1 | 2.i16-1.float | 123456.00 | 1.float-2.i16 | 0x2000 0x47F1 | +|0xF147 0x0020 | 2.i16-1.float-s | 123456.00 | 1.float-2.i16-s | 0xF147 0X0020 | +|0x0020 0xF147 | 2.i16-1.float-sb | 123456.00 | 1.float-2.i16-sb | 0x0020 0xF147 | +|0x47F1 0x2000 | 2.i16-1.float-sw | 123456.00 | 1.float-2.i16-sw | 0x47F1 0x2000 | + +Notice that different byte and word orderings require the use of the appropriate FieldServer function move. Once the proper function move is selected, the data can be converted in both directions. + +Of the many hex-to-floating point converters and calculators that are available in the Internet, very few actually allow manipulation of the byte and word orders. One such utility is located at www.61131.com/download.htm where both Linux and Windows versions of the utilities can be downloaded. Once installed, the utility is run as an executable with a single dialog interface. The utility presents the decimal float value of 123456.00 as follows: + +![Image5](img/img5.png) + +One can then swap bytes and/or words to analyze what potential endianness issues may exist between Modbus RTU master and slave devices. + +## Test +There are some programs in folder *test' to test the TCP communication: +- Ananas32/64 is slave simulator (only holding registers and inputs, no coils and digital inputs) +- RMMS is master simulator +- mod_RSsim.exe is slave simulator. It can be that you need [Microsoft Visual C++ 2008 SP1 Redistributable Package](https://www.microsoft.com/en-us/download/details.aspx?id=5582) to start it (because of SideBySide error). + +## Changelog +# 2.0.9 (2018-10-11) +* (Bjoern3003) Write registers was corrected + +# 2.0.7 (2018-07-02) +* (bluefox) The server mode was fixed + +# 2.0.6 (2018-06-26) +* (bluefox) rtu-tcp master mode was fixed + +# 2.0.3 (2018-06-16) +* (bluefox) Fixed the rounding of numbers + +# 2.0.2 (2018-06-12) +* (bluefox) The error with blocks reading was fixed +* (bluefox) The block reading for discrete values was implemented + +# 2.0.1 (2018-05-06) +* (bluefox) Added the support of multiple device IDs + +# 1.1.1 (2018-04-15) +* (Apollon77) Optimize reconnect handling + +# 1.1.0 (2018-01-23) +* (bluefox) Little endian strings added +* (Apollon77) Upgrade Serialport Library + +# 1.0.2 (2018-01-20) +* (bluefox) Fixed read of coils + +# 0.5.4 (2017-09-27) +* (Apollon77) Several Fixes + +# 0.5.0 (2017-02-11) +* (bluefox) Create all states each after other + +# 0.4.10 (2017-02-10) +* (Apollon77) Do not recreate all datapoints on start of adapter +* (ykuendig) Multiple optimization and wording fixes + +# 0.4.9 (2016-12-20) +* (bluefox) fix serial RTU + +# 0.4.8 (2016-12-15) +* (Apollon77) update serialport library for node 6.x compatibility + +# 0.4.7 (2016-11-27) +* (bluefox) Use old version of jsmodbus + +# 0.4.6 (2016-11-08) +* (bluefox) backward compatibility with 0.3.x + +# 0.4.5 (2016-10-25) +* (bluefox) better buffer handling on tcp and serial + +# 0.4.4 (2016-10-21) +* (bluefox) Fix write of holding registers + +# 0.4.1 (2016-10-19) +* (bluefox) Support of ModBus RTU over serial and over TCP (only slave) + +# 0.3.11 (2016-08-18) +* (Apollon77) Fix wrong byte count in loop + +# 0.3.10 (2016-02-01) +* (bluefox) fix lost of history settings. + +# 0.3.9 (2015-11-09) +* (bluefox) Use always write_multiple_registers by write of holding registers. + +# 0.3.7 (2015-11-02) +* (bluefox) add special read/write mode if "Max read request length" is 1. + +# 0.3.6 (2015-11-01) +* (bluefox) add cyclic write for holding registers (fix) + +# 0.3.5 (2015-10-31) +* (bluefox) add cyclic write for holding registers + +# 0.3.4 (2015-10-28) +* (bluefox) add doubles and fix uint64 + +# 0.3.3 (2015-10-27) +* (bluefox) fix holding registers + +# 0.3.2 (2015-10-27) +* (bluefox) fix import from text file + +# 0.3.1 (2015-10-26) +* (bluefox) fix error with length of read block (master) +* (bluefox) support of read blocks and maximal length of read request (master) +* (bluefox) can define fields by import + +# 0.3.0 (2015-10-24) +* (bluefox) add round settings +* (bluefox) add deviceID +* (bluefox) slave supports floats, integers and strings + +# 0.2.6 (2015-10-22) +* (bluefox) add different types for inputRegisters and for holding registers ONLY FOR MASTER + +# 0.2.5 (2015-10-20) +* (bluefox) fix names of objects if aliases used + +# 0.2.4 (2015-10-19) +* (bluefox) fix error add new values + +# 0.2.3 (2015-10-15) +* (bluefox) fix error with master + +# 0.2.2 (2015-10-14) +* (bluefox) implement slave +* (bluefox) change addressing model + +# 0.0.1 +* (bluefox) initial commit diff --git a/admin/i18n/de/translations.json b/admin/i18n/de/translations.json new file mode 100644 index 0000000..8f25574 --- /dev/null +++ b/admin/i18n/de/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Allgemein", + "Inputs": "Diskrete Eingänge", + "Do not align addresses to word:": "Die Adressen nicht auf 16 Bits ausrichten:", + "Coils": "Diskrete Ausgänge", + "Input Registers": "Eingangsregister", + "Holding Registers": "Eingangsregister", + "PLC Connection:": "SPS Verbindung:", + "PLC IP Address:": "SPS IP Adresse:", + "PLC Rack:": "SPS Rack:", + "PLC Slot:": "SPS Slot:", + "Round Real to:": "Aufrunden Real auf:", + "Poll delay:": "Poll delay:", + "deviceId": "Slave-ID", + "Reconnect time:": "Reconnect-Zeit:", + "Pulse time:": "Pulsetime:", + "Import symbols file:": "Symboldatei Importieren:", + "Import DB file:": "DB-Datei importieren:", + "Load Symbols": "Lade Symbole", + "Add DB": "DB einfügen", + "Toggle poll": "Poll umschalten", + "Toggle RW": "RW umschalten", + "Toggle WP": "WP umschalten", + "Address": "Adresse", + "Name": "Name", + "Description": "Beschreibung", + "Type": "Typ", + "Unit": "Einheit", + "poll": "poll", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Rolle", + "Room": "Raum", + "registers": "Register", + "Device ID:": "Geräte ID:", + "RTU over TCP": "RTU over TCP", + "Data bits:": "Data bits:", + "Stop bits:": "Stop bits:", + "Parity:": "Parity:", + "Read timeout:": "Read timeout:", + "Multi device IDs:": "Mehrere Geräte-IDs:", + "Use aliases as address:": "Aliases benutzen:", + "Max read request length:": "Maximale Lese-Request-Länge:", + "Enable polling of data point": "Zyklische Abfrage vom Datenpunkt", + "Write access allowed": "Schreiben erlaubt", + "Write pulses (true=>false edge)": "Schreibe Pulse (Ja=>Nein Flanke)", + "Connection parameters:": "Verbindungsparameter:", + "Partner IP Address:": "Partner IP Adresse:", + "Port:": "Port:", + "Type:": "Typ:", + "Master": "Master", + "Slave": "Slave", + "Are you sure?": "Sind Sie sicher?", + "Start address:": "Start-Adresse:", + "Text copied to clipboard. Click to close the window": "Text wurde in die Zwischenablage kopiert. Klicke um das Fenster zu schliessen.", + "Export": "Export", + "Import": "Import", + "Close": "Zumachen", + "Export to CSV": "Export in CSV", + "Import from CSV": "Import aus CSV", + "Delete all entries": "Alle Einträge löschen", + "Length": "Länge", + "All entries will be deleted. Are you sure?": "Alle Einträge werden gelöscht. Sind sie sicher?", + "Factor": "Faktor", + "Offset": "Offset", + "Cyclic write": "Zyklisch schreiben", + "TCP/Serial RTU:": "TCP/Serial RTU:", + "TCP": "TCP", + "Serial": "Serial", + "Baud rate:": "Baud rate:", + "Use direct addresses by aliases:": "Direkte Adressen benutzen (bei Aliases):", + "Select port": "Port wählen" +} \ No newline at end of file diff --git a/admin/i18n/en/translations.json b/admin/i18n/en/translations.json new file mode 100644 index 0000000..49ea21d --- /dev/null +++ b/admin/i18n/en/translations.json @@ -0,0 +1,74 @@ +{ + "General": "General", + "Inputs": "Discrete Inputs", + "Do not align addresses to word:": "Do not align addresses to 16 bits:", + "Coils": "Coils", + "Input Registers": "Input Registers", + "Holding Registers": "Holding Registers", + "PLC Connection:": "PLC Connection:", + "PLC IP Address:": "PLC IP Address:", + "PLC Rack:": "PLC Rack:", + "PLC Slot:": "PLC Slot:", + "Round Real to:": "Round real to:", + "Poll delay:": "Poll delay:", + "deviceId": "Slave ID", + "Reconnect time:": "Reconnect time:", + "Pulse time:": "Pulse time:", + "Import symbols file:": "Import symbols file:", + "Import DB file:": "Import DB file:", + "Load Symbols": "Load symbols", + "Add DB": "Add DB", + "Toggle poll": "Toggle poll", + "Toggle RW": "Toggle RW", + "Toggle WP": "Toggle WP", + "Address": "Address", + "Name": "Name", + "Description": "Description", + "Type": "Type", + "Unit": "Unit", + "poll": "poll", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Role", + "Room": "Room", + "registers": "registers", + "Device ID:": "Device ID:", + "RTU over TCP": "RTU over TCP", + "Data bits:": "Data bits:", + "Stop bits:": "Stop bits:", + "Parity:": "Parity:", + "Read timeout:": "Read timeout:", + "Multi device IDs:": "Multi device IDs:", + "Use aliases as address:": "Use aliases:", + "Max read request length:": "Max read request length:", + "Enable polling of data point": "Enable polling of data point", + "Write access allowed": "Write access allowed", + "Write pulses (true=>false edge)": "Write pulses (true=>false edge)", + "Connection parameters:": "Connection parameters:", + "Partner IP Address:": "Partner IP Address:", + "Port:": "Port:", + "Type:": "Type:", + "Master": "Master", + "Slave": "Slave", + "Are you sure?": "Are you sure?", + "Start address:": "Start address:", + "Text copied to clipboard. Click to close the window": "Text copied to clipboard. Click to close the window", + "Export": "Export", + "Import": "Import", + "Close": "Close", + "Export to CSV": "Export to CSV", + "Import from CSV": "Import from CSV", + "Delete all entries": "Delete all entries", + "Length": "Length", + "All entries will be deleted. Are you sure?": "All entries will be deleted. Are you sure?", + "Factor": "Factor", + "Offset": "Offset", + "Cyclic write": "Cyclic write", + "TCP/Serial RTU:": "TCP/Serial RTU:", + "TCP": "TCP", + "Serial": "Serial", + "Baud rate:": "Baud rate:", + "Use direct addresses by aliases:": "Use direct addresses by aliases:", + "Select port": "Select port" +} \ No newline at end of file diff --git a/admin/i18n/es/translations.json b/admin/i18n/es/translations.json new file mode 100644 index 0000000..5a888f7 --- /dev/null +++ b/admin/i18n/es/translations.json @@ -0,0 +1,74 @@ +{ + "General": "General", + "Inputs": "Entradas discretas", + "Do not align addresses to word:": "No alinee las direcciones a 16 bits:", + "Coils": "Bobinas", + "Input Registers": "Registros de entrada", + "Holding Registers": "Registros de mantenimiento", + "PLC Connection:": "Conexión de PLC:", + "PLC IP Address:": "Dirección IP del PLC:", + "PLC Rack:": "Estante del PLC:", + "PLC Slot:": "Ranura de PLC:", + "Round Real to:": "Redondo real para:", + "Poll delay:": "Retraso en la encuesta:", + "deviceId": "Slave ID", + "Reconnect time:": "Tiempo de reconexión:", + "Pulse time:": "Tiempo de pulso:", + "Import symbols file:": "Importar archivo de símbolos:", + "Import DB file:": "Importar archivo DB", + "Load Symbols": "Cargar símbolos", + "Add DB": "Agregar DB", + "Toggle poll": "Alternar encuesta", + "Toggle RW": "Alternar RW", + "Toggle WP": "Alternar WP", + "Address": "Dirección", + "Name": "Nombre", + "Description": "Descripción", + "Type": "Tipo", + "Unit": "Unidad", + "poll": "encuesta", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Papel", + "Room": "Habitación", + "registers": "registros", + "Device ID:": "ID del dispositivo:", + "RTU over TCP": "RTU sobre TCP", + "Data bits:": "Bits de datos:", + "Stop bits:": "Bits de parada:", + "Parity:": "Paridad:", + "Read timeout:": "Tiempo de espera de lectura:", + "Multi device IDs:": "ID de dispositivos múltiples:", + "Use aliases as address:": "Usa alias:", + "Max read request length:": "Longitud máxima de solicitud de lectura:", + "Enable polling of data point": "Habilitar el sondeo del punto de datos", + "Write access allowed": "Acceso de escritura permitido", + "Write pulses (true=>false edge)": "Escribir pulsos (verdadero => borde falso)", + "Connection parameters:": "Parámetros de conexión:", + "Partner IP Address:": "Dirección IP del socio:", + "Port:": "Puerto:", + "Type:": "Tipo:", + "Master": "Dominar", + "Slave": "Esclavo", + "Are you sure?": "¿Estás seguro?", + "Start address:": "Dirección de inicio:", + "Text copied to clipboard. Click to close the window": "Texto copiado al portapapeles. Haga clic para cerrar la ventana", + "Export": "Exportar", + "Import": "Importar", + "Close": "Cerca", + "Export to CSV": "Exportar a CSV", + "Import from CSV": "Importar desde CSV", + "Delete all entries": "Eliminar todas las entradas", + "Length": "Longitud", + "All entries will be deleted. Are you sure?": "Todas las entradas serán eliminadas. ¿Estás seguro?", + "Factor": "Factor", + "Offset": "Compensar", + "Cyclic write": "Escritura cíclica", + "TCP/Serial RTU:": "TCP / Serial RTU:", + "TCP": "TCP", + "Serial": "De serie", + "Baud rate:": "Velocidad de baudios:", + "Use direct addresses by aliases:": "Use direcciones directas por alias:", + "Select port": "Seleccionar puerto" +} \ No newline at end of file diff --git a/admin/i18n/fr/translations.json b/admin/i18n/fr/translations.json new file mode 100644 index 0000000..67465d9 --- /dev/null +++ b/admin/i18n/fr/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Général", + "Inputs": "Entrées discrètes", + "Do not align addresses to word:": "Ne pas aligner les adresses sur 16 bits:", + "Coils": "Bobines", + "Input Registers": "Registres d'entrée", + "Holding Registers": "Tenir des registres", + "PLC Connection:": "Connexion PLC:", + "PLC IP Address:": "Adresse IP de l'API:", + "PLC Rack:": "Rack PLC:", + "PLC Slot:": "Slot PLC:", + "Round Real to:": "Rond réel à:", + "Poll delay:": "Délai d'interrogation:", + "deviceId": "Slave ID", + "Reconnect time:": "Reconnectez le temps:", + "Pulse time:": "Temps d'impulsion:", + "Import symbols file:": "Importer un fichier de symboles:", + "Import DB file:": "Importer un fichier DB:", + "Load Symbols": "Charger des symboles", + "Add DB": "Ajouter une DB", + "Toggle poll": "Basculer le sondage", + "Toggle RW": "Toggle RW", + "Toggle WP": "Toggle WP", + "Address": "Adresse", + "Name": "prénom", + "Description": "La description", + "Type": "Type", + "Unit": "Unité", + "poll": "sondage", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Rôle", + "Room": "Chambre", + "registers": "registres", + "Device ID:": "Reference de l'appareil:", + "RTU over TCP": "RTU sur TCP", + "Data bits:": "Bits de données:", + "Stop bits:": "Bits d'arrêt:", + "Parity:": "Parité:", + "Read timeout:": "Lire le délai d'attente:", + "Multi device IDs:": "ID multi-appareils:", + "Use aliases as address:": "Utiliser des alias:", + "Max read request length:": "Max longueur de la requête de lecture:", + "Enable polling of data point": "Activer l'interrogation du point de données", + "Write access allowed": "Accès en écriture autorisé", + "Write pulses (true=>false edge)": "Écrire des impulsions (true => false edge)", + "Connection parameters:": "Paramètres de connexion:", + "Partner IP Address:": "Adresse IP du partenaire:", + "Port:": "Port:", + "Type:": "Type:", + "Master": "Maîtriser", + "Slave": "Esclave", + "Are you sure?": "Êtes-vous sûr?", + "Start address:": "Adresse de départ:", + "Text copied to clipboard. Click to close the window": "Texte copié dans le presse-papier Cliquez pour fermer la fenêtre", + "Export": "Exportation", + "Import": "Importer", + "Close": "Fermer", + "Export to CSV": "Exporter au format CSV", + "Import from CSV": "Importer à partir du fichier CSV", + "Delete all entries": "Supprimer toutes les entrées", + "Length": "Longueur", + "All entries will be deleted. Are you sure?": "Toutes les entrées seront supprimées. Êtes-vous sûr?", + "Factor": "Facteur", + "Offset": "Décalage", + "Cyclic write": "Écriture cyclique", + "TCP/Serial RTU:": "TCP / Sériel RTU:", + "TCP": "TCP", + "Serial": "En série", + "Baud rate:": "Débit en bauds:", + "Use direct addresses by aliases:": "Utilisez des adresses directes par alias:", + "Select port": "Sélectionnez un port" +} \ No newline at end of file diff --git a/admin/i18n/it/translations.json b/admin/i18n/it/translations.json new file mode 100644 index 0000000..82012c3 --- /dev/null +++ b/admin/i18n/it/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Generale", + "Inputs": "Ingressi discreti", + "Do not align addresses to word:": "Non allineare gli indirizzi a 16 bit:", + "Coils": "bobine", + "Input Registers": "Registri di input", + "Holding Registers": "Holding Registers", + "PLC Connection:": "Connessione PLC:", + "PLC IP Address:": "Indirizzo IP del PLC:", + "PLC Rack:": "Rack PLC:", + "PLC Slot:": "Slot PLC:", + "Round Real to:": "Round reale a:", + "Poll delay:": "Ritardo del sondaggio:", + "deviceId": "ID slave", + "Reconnect time:": "Tempo di riconnessione:", + "Pulse time:": "Tempo di impulso:", + "Import symbols file:": "Importa file simboli:", + "Import DB file:": "Importa file DB:", + "Load Symbols": "Carica simboli", + "Add DB": "Aggiungi DB", + "Toggle poll": "Attiva / disattiva sondaggio", + "Toggle RW": "Attiva / disattiva RW", + "Toggle WP": "Attiva / disattiva WP", + "Address": "Indirizzo", + "Name": "Nome", + "Description": "Descrizione", + "Type": "genere", + "Unit": "Unità", + "poll": "sondaggio", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Ruolo", + "Room": "Camera", + "registers": "registri", + "Device ID:": "ID del dispositivo:", + "RTU over TCP": "RTU su TCP", + "Data bits:": "Bit di dati:", + "Stop bits:": "Stop bit:", + "Parity:": "Parità:", + "Read timeout:": "Leggi il timeout:", + "Multi device IDs:": "ID multi dispositivo:", + "Use aliases as address:": "Usa alias:", + "Max read request length:": "Lunghezza massima richiesta di lettura:", + "Enable polling of data point": "Abilita il polling del punto dati", + "Write access allowed": "Accesso in scrittura consentito", + "Write pulses (true=>false edge)": "Scrivi impulsi (true => falso bordo)", + "Connection parameters:": "Parametri di connessione:", + "Partner IP Address:": "Indirizzo IP del partner:", + "Port:": "Porta:", + "Type:": "Genere:", + "Master": "Maestro", + "Slave": "Schiavo", + "Are you sure?": "Sei sicuro?", + "Start address:": "Indirizzo iniziale:", + "Text copied to clipboard. Click to close the window": "Testo copiato negli appunti. Clicca per chiudere la finestra", + "Export": "Esportare", + "Import": "Importare", + "Close": "Vicino", + "Export to CSV": "Esporta in CSV", + "Import from CSV": "Importa da CSV", + "Delete all entries": "Elimina tutte le voci", + "Length": "Lunghezza", + "All entries will be deleted. Are you sure?": "Tutte le voci saranno cancellate. Sei sicuro?", + "Factor": "Fattore", + "Offset": "Compensare", + "Cyclic write": "Scrittura ciclica", + "TCP/Serial RTU:": "RTU TCP / seriale:", + "TCP": "TCP", + "Serial": "Seriale", + "Baud rate:": "Baud rate:", + "Use direct addresses by aliases:": "Usa indirizzi diretti per alias:", + "Select port": "Seleziona porta" +} \ No newline at end of file diff --git a/admin/i18n/nl/translations.json b/admin/i18n/nl/translations.json new file mode 100644 index 0000000..f084d5c --- /dev/null +++ b/admin/i18n/nl/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Algemeen", + "Inputs": "Discrete ingangen", + "Do not align addresses to word:": "Lijn adressen niet uit met 16 bits:", + "Coils": "coils", + "Input Registers": "Invoegregisters", + "Holding Registers": "Registers houden", + "PLC Connection:": "PLC-verbinding:", + "PLC IP Address:": "PLC IP-adres:", + "PLC Rack:": "PLC Rack:", + "PLC Slot:": "PLC-slot:", + "Round Real to:": "Rond echt naar:", + "Poll delay:": "Poll vertraging:", + "deviceId": "Slave ID", + "Reconnect time:": "Reconnect tijd:", + "Pulse time:": "Pulstijd:", + "Import symbols file:": "Symboolbestand importeren:", + "Import DB file:": "DB-bestand importeren:", + "Load Symbols": "Laad symbolen", + "Add DB": "Voeg DB toe", + "Toggle poll": "Poll wisselen", + "Toggle RW": "RW omschakelen", + "Toggle WP": "Wissel WP", + "Address": "Adres", + "Name": "Naam", + "Description": "Beschrijving", + "Type": "Type", + "Unit": "Eenheid", + "poll": "poll", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Rol", + "Room": "Kamer", + "registers": "registers", + "Device ID:": "Apparaat ID:", + "RTU over TCP": "RTU via TCP", + "Data bits:": "Databits:", + "Stop bits:": "Stop bits:", + "Parity:": "Pariteit:", + "Read timeout:": "Lees time-out:", + "Multi device IDs:": "ID's voor meerdere apparaten:", + "Use aliases as address:": "Gebruik aliassen:", + "Max read request length:": "Max. Leesverzoeklengte:", + "Enable polling of data point": "Polling van gegevenspunt inschakelen", + "Write access allowed": "Schrijftoegang toegestaan", + "Write pulses (true=>false edge)": "Schrijf pulsen (true => false edge)", + "Connection parameters:": "Verbindingsparameters:", + "Partner IP Address:": "IP-adres van partner:", + "Port:": "Haven:", + "Type:": "Type:", + "Master": "Meester", + "Slave": "Slaaf", + "Are you sure?": "Weet je het zeker?", + "Start address:": "Start adres:", + "Text copied to clipboard. Click to close the window": "Tekst gekopieerd naar klembord. Klik om het venster te sluiten", + "Export": "Exporteren", + "Import": "Importeren", + "Close": "Dichtbij", + "Export to CSV": "Exporteren naar CSV", + "Import from CSV": "Importeren vanuit CSV", + "Delete all entries": "Verwijder alle vermeldingen", + "Length": "Lengte", + "All entries will be deleted. Are you sure?": "Alle inzendingen worden verwijderd. Weet je het zeker?", + "Factor": "Factor", + "Offset": "compenseren", + "Cyclic write": "Cyclisch schrijven", + "TCP/Serial RTU:": "TCP / Serial RTU:", + "TCP": "TCP", + "Serial": "serie-", + "Baud rate:": "Baudrate:", + "Use direct addresses by aliases:": "Gebruik directe adressen op aliassen:", + "Select port": "Selecteer poort" +} \ No newline at end of file diff --git a/admin/i18n/pl/translations.json b/admin/i18n/pl/translations.json new file mode 100644 index 0000000..7ba243f --- /dev/null +++ b/admin/i18n/pl/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Generał", + "Inputs": "Wejścia dyskretne", + "Do not align addresses to word:": "Nie wyrównaj adresów do 16 bitów:", + "Coils": "Cewki", + "Input Registers": "Rejestry wejściowe", + "Holding Registers": "Prowadzenie rejestrów", + "PLC Connection:": "Połączenie PLC:", + "PLC IP Address:": "Adres IP sterownika PLC:", + "PLC Rack:": "Rack PLC:", + "PLC Slot:": "Gniazdo PLC:", + "Round Real to:": "Runda prawdziwa do:", + "Poll delay:": "Opóźnienie ankiety:", + "deviceId": "ID Slave", + "Reconnect time:": "Czas ponownego połączenia:", + "Pulse time:": "Czas impulsu:", + "Import symbols file:": "Importuj plik symboli:", + "Import DB file:": "Importuj plik DB:", + "Load Symbols": "Załaduj symbole", + "Add DB": "Dodaj DB", + "Toggle poll": "Przełącz ankietę", + "Toggle RW": "Przełącz RW", + "Toggle WP": "Przełącz WP", + "Address": "Adres", + "Name": "Nazwa", + "Description": "Opis", + "Type": "Rodzaj", + "Unit": "Jednostka", + "poll": "głosowanie", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Rola", + "Room": "Pokój", + "registers": "rejestry", + "Device ID:": "Identyfikator urzadzenia:", + "RTU over TCP": "RTU przez TCP", + "Data bits:": "Bity danych:", + "Stop bits:": "Stop bitów:", + "Parity:": "Parytet:", + "Read timeout:": "Odczyt limitu czasu:", + "Multi device IDs:": "Identyfikatory wielu urządzeń:", + "Use aliases as address:": "Użyj aliasów:", + "Max read request length:": "Maksymalna długość żądania odczytu:", + "Enable polling of data point": "Włącz odpytywanie punktu danych", + "Write access allowed": "Dostęp do zapisu dozwolony", + "Write pulses (true=>false edge)": "Pisz impulsy (true => false edge)", + "Connection parameters:": "Parametry połączenia:", + "Partner IP Address:": "Adres IP partnera:", + "Port:": "Port:", + "Type:": "Rodzaj:", + "Master": "Mistrz", + "Slave": "Niewolnik", + "Are you sure?": "Jesteś pewny?", + "Start address:": "Adres początkowy:", + "Text copied to clipboard. Click to close the window": "Tekst skopiowany do schowka. Kliknij, aby zamknąć okno", + "Export": "Eksport", + "Import": "Import", + "Close": "Blisko", + "Export to CSV": "Eksportuj do pliku CSV", + "Import from CSV": "Importuj z CSV", + "Delete all entries": "Usuń wszystkie wpisy", + "Length": "Długość", + "All entries will be deleted. Are you sure?": "Wszystkie wpisy zostaną usunięte. Jesteś pewny?", + "Factor": "Czynnik", + "Offset": "Offsetowy", + "Cyclic write": "Cykliczny zapis", + "TCP/Serial RTU:": "TCP / Serial RTU:", + "TCP": "TCP", + "Serial": "Seryjny", + "Baud rate:": "Szybkość transmisji:", + "Use direct addresses by aliases:": "Użyj bezpośrednich adresów przez aliasy:", + "Select port": "Wybierz port" +} \ No newline at end of file diff --git a/admin/i18n/pt/translations.json b/admin/i18n/pt/translations.json new file mode 100644 index 0000000..bfae85a --- /dev/null +++ b/admin/i18n/pt/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Geral", + "Inputs": "Entradas discretas", + "Do not align addresses to word:": "Não alinhe endereços para 16 bits:", + "Coils": "Bobinas", + "Input Registers": "Registos de entrada", + "Holding Registers": "Registros de retenção", + "PLC Connection:": "Conexão do PLC:", + "PLC IP Address:": "Endereço IP do PLC:", + "PLC Rack:": "PLC Rack:", + "PLC Slot:": "Slot PLC:", + "Round Real to:": "Round real to:", + "Poll delay:": "Retardo de enquete:", + "deviceId": "Slave ID", + "Reconnect time:": "Reconectar o tempo:", + "Pulse time:": "Tempo de pulso:", + "Import symbols file:": "Arquivo de símbolos de importação:", + "Import DB file:": "Importar arquivo DB:", + "Load Symbols": "Carregar símbolos", + "Add DB": "Adicionar DB", + "Toggle poll": "Alternar pesquisa", + "Toggle RW": "Toggle RW", + "Toggle WP": "Toggle WP", + "Address": "Endereço", + "Name": "Nome", + "Description": "Descrição", + "Type": "Tipo", + "Unit": "Unidade", + "poll": "votação", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Função", + "Room": "Quarto", + "registers": "registros", + "Device ID:": "ID de dispositivo:", + "RTU over TCP": "RTU sobre TCP", + "Data bits:": "Bits de dados:", + "Stop bits:": "Parar bits:", + "Parity:": "Paridade:", + "Read timeout:": "Tempo de ler esgotado:", + "Multi device IDs:": "IDs de vários dispositivos:", + "Use aliases as address:": "Use aliases:", + "Max read request length:": "Comprimento máximo do pedido de leitura:", + "Enable polling of data point": "Habilitar a votação do ponto de dados", + "Write access allowed": "Acesso de acesso permitido", + "Write pulses (true=>false edge)": "Escrever pulsos (true => false edge)", + "Connection parameters:": "Parâmetros de conexão:", + "Partner IP Address:": "Endereço IP do parceiro:", + "Port:": "Porta:", + "Type:": "Tipo:", + "Master": "mestre", + "Slave": "Escravo", + "Are you sure?": "Você tem certeza?", + "Start address:": "Endereço de início:", + "Text copied to clipboard. Click to close the window": "Texto copiado para a área de transferência. Clique para fechar a janela", + "Export": "Exportar", + "Import": "Importar", + "Close": "Fechar", + "Export to CSV": "Exportar para CSV", + "Import from CSV": "Importação de CSV", + "Delete all entries": "Eliminar todas as entradas", + "Length": "comprimento", + "All entries will be deleted. Are you sure?": "Todas as entradas serão excluídas. Você tem certeza?", + "Factor": "Fator", + "Offset": "Offset", + "Cyclic write": "Escrita cíclica", + "TCP/Serial RTU:": "TCP / Serial RTU:", + "TCP": "TCP", + "Serial": "Serial", + "Baud rate:": "Taxa de transmissão:", + "Use direct addresses by aliases:": "Use endereços diretos por alias:", + "Select port": "Selecione a porta" +} \ No newline at end of file diff --git a/admin/i18n/ru/translations.json b/admin/i18n/ru/translations.json new file mode 100644 index 0000000..f09929e --- /dev/null +++ b/admin/i18n/ru/translations.json @@ -0,0 +1,74 @@ +{ + "General": "Основное", + "Inputs": "Дискретные входы", + "Do not align addresses to word:": "Не выравнивать адреса до 16 бит:", + "Coils": "Регистры флагов", + "Input Registers": "Регистры входа", + "Holding Registers": "Регистры хранения", + "PLC Connection:": "PLC соединение:", + "PLC IP Address:": "PLC IP адрес:", + "PLC Rack:": "PLC Rack:", + "PLC Slot:": "PLC слот:", + "Round Real to:": "Округлять real до:", + "Poll delay:": "Интервал опроса:", + "deviceId": "Slave ID", + "Reconnect time:": "Reconnect time:", + "Pulse time:": "Pulse time:", + "Import symbols file:": "Ипморт символьных файлов:", + "Import DB file:": "Ипморт DB файлов:", + "Load Symbols": "Загрузить символы", + "Add DB": "Добавить DB", + "Toggle poll": "Изменить poll", + "Toggle RW": "Изменить RW", + "Toggle WP": "Изменить WP", + "Address": "Адрес", + "Name": "Имя", + "Description": "Описание", + "Type": "Тип", + "Unit": "Единицы", + "poll": "poll", + "RW": "RW", + "CW": "CW", + "WP": "WP", + "Role": "Роль", + "Room": "Комната", + "registers": "регистров", + "Device ID:": "ID устройства:", + "RTU over TCP": "RTU over TCP", + "Data bits:": "Data bits:", + "Stop bits:": "Stop bits:", + "Parity:": "Parity:", + "Read timeout:": "Таймаут чтения:", + "Multi device IDs:": "Несколько ID:", + "Use aliases as address:": "Использовать alias:", + "Max read request length:": "Макс. кол-во регистров при чтении:", + "Enable polling of data point": "Постоянный опрос переменной в каждом цикле", + "Write access allowed": "Разрешить запись в переменную", + "Write pulses (true=>false edge)": "Генерировать импульсы (1 => 0)", + "Connection parameters:": "Параметры соединения:", + "Partner IP Address:": "IP адрес партнёра:", + "Port:": "Порт:", + "Type:": "Тип:", + "Master": "Master", + "Slave": "Slave", + "Are you sure?": "Вы уверенны?", + "Start address:": "Начальный адрес:", + "Text copied to clipboard. Click to close the window": "Текст скопирован в буфер обмена. Щелкните мышкой здесь, чтобы закрыть окно", + "Export": "Экспорт", + "Import": "Импорт", + "Close": "Закрыть", + "Export to CSV": "Экспорт в CSV", + "Import from CSV": "Импорт из CSV", + "Delete all entries": "Удалить все элементы", + "Length": "Длина", + "All entries will be deleted. Are you sure?": "Все элементы будут удалены. Вы уверенны?", + "Factor": "Множитель", + "Offset": "Сдвиг", + "Cyclic write": "Писать в каждом цикле", + "TCP/Serial RTU:": "TCP/Serial RTU:", + "TCP": "TCP", + "Serial": "Serial", + "Baud rate:": "Скорость порта:", + "Use direct addresses by aliases:": "Использовать прямые адреса при alias:", + "Select port": "Выберите порт" +} \ No newline at end of file diff --git a/admin/img/plc_back.png b/admin/img/plc_back.png new file mode 100644 index 0000000000000000000000000000000000000000..9b03ce05d3c0b841efcbbf30b3580471f36e1d0a GIT binary patch literal 68765 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 R^DF=W002ovPDHLkV1iuCj$8l$ literal 0 HcmV?d00001 diff --git a/admin/index.html b/admin/index.html new file mode 100644 index 0000000..0e1d169 --- /dev/null +++ b/admin/index.html @@ -0,0 +1,1372 @@ + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Connection parameters:
+ +
+
+
+
+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
General
ms
ms
ms
ms
registers
registers
+
+
+
+
Start address:
+ + + +
+
+
+
+
+
Start address:
+ + + + + +
+
+
+
+
+
Start address:
+ + + +
+
+
+
+
+
Start address:
+ + + + + +
+
+
+ + + +
+ + diff --git a/admin/lib/css/jsgrid-theme.css b/admin/lib/css/jsgrid-theme.css new file mode 100644 index 0000000..936950b --- /dev/null +++ b/admin/lib/css/jsgrid-theme.css @@ -0,0 +1,205 @@ +/* + * jsGrid v1.0.1 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid-grid-header, +.jsgrid-grid-body, +.jsgrid-header-row > th, +.jsgrid-filter-row > td, +.jsgrid-insert-row > td, +.jsgrid-edit-row > td { + border: 1px solid #e9e9e9; +} + +.jsgrid-header-row > th { + border-top: 0; +} + +.jsgrid-header-row > th, .jsgrid-filter-row > td, .jsgrid-insert-row > td { + border-bottom: 0; +} + +.jsgrid-header-row > th:first-child, .jsgrid-filter-row > td:first-child, .jsgrid-insert-row > td:first-child { + border-left: none; +} + +.jsgrid-grid-header { + background: #f9f9f9; +} + +.jsgrid-header-sortable:hover { + cursor: pointer; + background: #fcfcfc; +} + +.jsgrid-header-row .jsgrid-header-sort { + background: #c4e2ff; +} + +.jsgrid-header-sort:before { + content: " "; + display: block; + float: left; + width: 0; + height: 0; + border-style: solid; +} + +.jsgrid-header-sort-asc:before { + border-width: 0 5px 5px 5px; + border-color: transparent transparent #009a67 transparent; +} + +.jsgrid-header-sort-desc:before { + border-width: 5px 5px 0 5px; + border-color: #009a67 transparent transparent transparent; +} + +.jsgrid-grid-body { + border-top: none; +} + +.jsgrid-grid-body td { + border: #f3f3f3 1px solid; +} + +.jsgrid-grid-body tr:first-child td { + border-top: none; +} + +.jsgrid-grid-body tr td:first-child { + border-left: none; +} + +.jsgrid-row > td { + background: #fff; +} + +.jsgrid-alt-row > td { + background: #fcfcfc; +} + +.jsgrid-header-row > th { + background: #f9f9f9; +} + +.jsgrid-filter-row > td { + background: #fcfcfc; +} + +.jsgrid-insert-row > td { + background: #e3ffe5; +} + +.jsgrid-edit-row > td { + background: #fdffe3; +} + +.jsgrid-selected-row > td { + background: #c4e2ff; + border-color: #c4e2ff; +} + +.jsgrid-nodata-row td { + background: #fff; +} + +.jsgrid-pager-current-page { + font-weight: bold; +} + +.jsgrid-button + .jsgrid-button { + margin-left: 5px; +} + +.jsgrid-button:hover { + opacity: .5; + transition: opacity 200ms linear; +} + +.jsgrid .jsgrid-button { + width: 16px; + height: 16px; + border: none; + cursor: pointer; + background-image: url(); + background-repeat: no-repeat; + background-color: transparent; +} + +@media only screen and (-webkit-min-device-pixel-ratio: 2), only screen and (min-device-pixel-ratio: 2) { + .jsgrid .jsgrid-button { + background-image: url(); + background-size: 24px 352px; + } +} + +.jsgrid .jsgrid-mode-button { + width: 24px; + height: 24px; +} + +.jsgrid-cancel-button { background-position: 0 0; width: 16px; height: 16px; } +.jsgrid-clear-filter-button { background-position: 0 -40px; width: 16px; height: 16px; } +.jsgrid-delete-button { background-position: 0 -80px; width: 16px; height: 16px; } +.jsgrid-edit-button { background-position: 0 -120px; width: 16px; height: 16px; } +.jsgrid-insert-mode-button { background-position: 0 -160px; width: 24px; height: 24px; } +.jsgrid-insert-button { background-position: 0 -208px; width: 16px; height: 16px; } +.jsgrid-search-mode-button { background-position: 0 -248px; width: 24px; height: 24px; } +.jsgrid-search-button { background-position: 0 -296px; width: 16px; height: 16px; } +.jsgrid-update-button { background-position: 0 -336px; width: 16px; height: 16px; } + + +.jsgrid-load-shader { + background: #ddd; + opacity: .5; + filter: alpha(opacity=50); +} + +.jsgrid-load-panel { + width: 15em; + height: 5em; + background: #fff; + border: 1px solid #e9e9e9; + padding-top: 3em; + text-align: center; +} + +.jsgrid-load-panel:before { + content: ' '; + position: absolute; + top: .5em; + left: 50%; + margin-left: -1em; + width: 2em; + height: 2em; + border: 2px solid #009a67; + border-right-color: transparent; + border-radius: 50%; + -webkit-animation: indicator 1s linear infinite; + animation: indicator 1s linear infinite; +} + +@-webkit-keyframes indicator +{ + from { -webkit-transform: rotate(0deg); } + 50% { -webkit-transform: rotate(180deg); } + to { -webkit-transform: rotate(360deg); } +} + +@keyframes indicator +{ + from { transform: rotate(0deg); } + 50% { transform: rotate(180deg); } + to { transform: rotate(360deg); } +} + +/* old IE */ +.jsgrid-load-panel { + padding-top: 1.5em \ ; +} +.jsgrid-load-panel:before { + display: none \ ; +} diff --git a/admin/lib/css/jsgrid-theme.min.css b/admin/lib/css/jsgrid-theme.min.css new file mode 100644 index 0000000..31e9c8c --- /dev/null +++ b/admin/lib/css/jsgrid-theme.min.css @@ -0,0 +1,7 @@ +/* + * jsGrid v1.0.1 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid-edit-row>td,.jsgrid-filter-row>td,.jsgrid-grid-body,.jsgrid-grid-header,.jsgrid-header-row>th,.jsgrid-insert-row>td{border:1px solid #e9e9e9}.jsgrid-header-row>th{border-top:0}.jsgrid-filter-row>td,.jsgrid-header-row>th,.jsgrid-insert-row>td{border-bottom:0}.jsgrid-filter-row>td:first-child,.jsgrid-header-row>th:first-child,.jsgrid-insert-row>td:first-child{border-left:none}.jsgrid-grid-header{background:#f9f9f9}.jsgrid-header-sortable:hover{cursor:pointer;background:#fcfcfc}.jsgrid-header-row .jsgrid-header-sort{background:#c4e2ff}.jsgrid-header-sort:before{content:" ";display:block;float:left;width:0;height:0;border-style:solid}.jsgrid-header-sort-asc:before{border-width:0 5px 5px;border-color:transparent transparent #009a67}.jsgrid-header-sort-desc:before{border-width:5px 5px 0;border-color:#009a67 transparent transparent}.jsgrid-grid-body{border-top:none}.jsgrid-grid-body td{border:1px solid #f3f3f3}.jsgrid-grid-body tr:first-child td{border-top:none}.jsgrid-grid-body tr td:first-child{border-left:none}.jsgrid-row>td{background:#fff}.jsgrid-alt-row>td{background:#fcfcfc}.jsgrid-header-row>th{background:#f9f9f9}.jsgrid-filter-row>td{background:#fcfcfc}.jsgrid-insert-row>td{background:#e3ffe5}.jsgrid-edit-row>td{background:#fdffe3}.jsgrid-selected-row>td{background:#c4e2ff;border-color:#c4e2ff}.jsgrid-nodata-row td{background:#fff}.jsgrid-pager-current-page{font-weight:700}.jsgrid-button+.jsgrid-button{margin-left:5px}.jsgrid-button:hover{opacity:.5;transition:opacity 200ms linear}.jsgrid .jsgrid-button{width:16px;height:16px;border:none;cursor:pointer;background-image:url();background-repeat:no-repeat;background-color:transparent}@media only screen and (-webkit-min-device-pixel-ratio:2),only screen and (min-device-pixel-ratio:2){.jsgrid .jsgrid-button{background-image:url();background-size:24px 352px}}.jsgrid .jsgrid-mode-button{width:24px;height:24px}.jsgrid-cancel-button{background-position:0 0;width:16px;height:16px}.jsgrid-clear-filter-button{background-position:0 -40px;width:16px;height:16px}.jsgrid-delete-button{background-position:0 -80px;width:16px;height:16px}.jsgrid-edit-button{background-position:0 -120px;width:16px;height:16px}.jsgrid-insert-mode-button{background-position:0 -160px;width:24px;height:24px}.jsgrid-insert-button{background-position:0 -208px;width:16px;height:16px}.jsgrid-search-mode-button{background-position:0 -248px;width:24px;height:24px}.jsgrid-search-button{background-position:0 -296px;width:16px;height:16px}.jsgrid-update-button{background-position:0 -336px;width:16px;height:16px}.jsgrid-load-shader{background:#ddd;opacity:.5;filter:alpha(opacity=50)}.jsgrid-load-panel{width:15em;height:5em;background:#fff;border:1px solid #e9e9e9;padding-top:3em;text-align:center}.jsgrid-load-panel:before{content:' ';position:absolute;top:.5em;left:50%;margin-left:-1em;width:2em;height:2em;border:2px solid #009a67;border-right-color:transparent;border-radius:50%;-webkit-animation:indicator 1s linear infinite;animation:indicator 1s linear infinite}@-webkit-keyframes indicator{from{-webkit-transform:rotate(0deg)}50%{-webkit-transform:rotate(180deg)}to{-webkit-transform:rotate(360deg)}}@keyframes indicator{from{transform:rotate(0deg)}50%{transform:rotate(180deg)}to{transform:rotate(360deg)}}.jsgrid-load-panel{padding-top:1.5em \}.jsgrid-load-panel:before{display:none \} \ No newline at end of file diff --git a/admin/lib/css/jsgrid.css b/admin/lib/css/jsgrid.css new file mode 100644 index 0000000..33fc780 --- /dev/null +++ b/admin/lib/css/jsgrid.css @@ -0,0 +1,121 @@ +/* + * jsGrid v1.0.1 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid { + position: relative; + overflow: hidden; + font-size: 1em; +} + +.jsgrid, .jsgrid *, .jsgrid *:before, .jsgrid *:after { + box-sizing: border-box; +} + +.jsgrid input, +.jsgrid textarea, +.jsgrid select { + font-size: 1em; +} + +.jsgrid-grid-header { + overflow-x: hidden; + overflow-y: hidden; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + -o-user-select: none; + user-select: none; +} + +.jsgrid-grid-body { + overflow-x: hidden; + overflow-y: scroll; + -webkit-overflow-scrolling: touch; +} + +.jsgrid-table { + width: 100%; + table-layout: fixed; + border-collapse: collapse; + border-spacing: 0; +} + +.jsgrid-table td { + padding: 0.5em 0.5em; +} + +.jsgrid-table td, +.jsgrid-table th { + box-sizing: border-box; +} + +.jsgrid-align-left { + text-align: left; +} + +.jsgrid-align-center { + text-align: center; +} + +.jsgrid-align-right { + text-align: right; +} + +.jsgrid-header-row > th { + padding: .5em .5em; +} + +.jsgrid-filter-row input, +.jsgrid-filter-row textarea, +.jsgrid-filter-row select, +.jsgrid-edit-row input, +.jsgrid-edit-row textarea, +.jsgrid-edit-row select, +.jsgrid-insert-row input, +.jsgrid-insert-row textarea, +.jsgrid-insert-row select { + width: 90%; + padding: .3em .5em; +} + +.jsgrid-filter-row input[type='checkbox'], +.jsgrid-edit-row input[type='checkbox'], +.jsgrid-insert-row input[type='checkbox'] { + width: auto; +} + +.jsgrid-header-row > th, +.jsgrid-filter-row > td, +.jsgrid-insert-row > td, +.jsgrid-edit-row > td { + text-align: center; +} + +.jsgrid-selected-row td { + cursor: pointer; +} + +.jsgrid-nodata-row td { + padding: .5em 0; + text-align: center; +} + +.jsgrid-header-sort { + cursor: pointer; +} + +.jsgrid-pager { + padding: .5em 0; +} + +.jsgrid-pager-nav-button { + padding: .2em .6em; +} + +.jsgrid-pager-page { + padding: .2em .6em; +} \ No newline at end of file diff --git a/admin/lib/css/jsgrid.min.css b/admin/lib/css/jsgrid.min.css new file mode 100644 index 0000000..6463bb9 --- /dev/null +++ b/admin/lib/css/jsgrid.min.css @@ -0,0 +1,7 @@ +/* + * jsGrid v1.0.1 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +.jsgrid{position:relative;overflow:hidden;font-size:1em}.jsgrid,.jsgrid *,.jsgrid :after,.jsgrid :before{box-sizing:border-box}.jsgrid input,.jsgrid select,.jsgrid textarea{font-size:1em}.jsgrid-grid-header{overflow-x:hidden;overflow-y:hidden;-webkit-user-select:none;-khtml-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;user-select:none}.jsgrid-grid-body{overflow-x:hidden;overflow-y:scroll;-webkit-overflow-scrolling:touch}.jsgrid-table{width:100%;table-layout:fixed;border-collapse:collapse;border-spacing:0}.jsgrid-table td{padding:.5em}.jsgrid-table td,.jsgrid-table th{box-sizing:border-box}.jsgrid-align-left{text-align:left}.jsgrid-align-center{text-align:center}.jsgrid-align-right{text-align:right}.jsgrid-header-row>th{padding:.5em}.jsgrid-edit-row input,.jsgrid-edit-row select,.jsgrid-edit-row textarea,.jsgrid-filter-row input,.jsgrid-filter-row select,.jsgrid-filter-row textarea,.jsgrid-insert-row input,.jsgrid-insert-row select,.jsgrid-insert-row textarea{width:90%;padding:.3em .5em}.jsgrid-edit-row input[type=checkbox],.jsgrid-filter-row input[type=checkbox],.jsgrid-insert-row input[type=checkbox]{width:auto}.jsgrid-edit-row>td,.jsgrid-filter-row>td,.jsgrid-header-row>th,.jsgrid-insert-row>td{text-align:center}.jsgrid-selected-row td{cursor:pointer}.jsgrid-nodata-row td{padding:.5em 0;text-align:center}.jsgrid-header-sort{cursor:pointer}.jsgrid-pager{padding:.5em 0}.jsgrid-pager-nav-button,.jsgrid-pager-page{padding:.2em .6em} \ No newline at end of file diff --git a/admin/lib/js/grid.locale-de.js b/admin/lib/js/grid.locale-de.js new file mode 100644 index 0000000..9c07a68 --- /dev/null +++ b/admin/lib/js/grid.locale-de.js @@ -0,0 +1,10 @@ +function translate_de() { + window.jsGrid.Grid.prototype.noDataContent = "Nicht gefunden"; + window.jsGrid.Grid.prototype.deleteConfirm = "Sind Sie sicher?"; + window.jsGrid.Grid.prototype.pagerFormat = "Seiten = {first} {prev} {pages} {next} {last}    {pageIndex} of {pageCount}"; + window.jsGrid.Grid.prototype.pagePrevText = "Vorh."; + window.jsGrid.Grid.prototype.pageNextText = "Nexte"; + window.jsGrid.Grid.prototype.pageFirstText = "Erste"; + window.jsGrid.Grid.prototype.pageLastText = "Letzte"; + window.jsGrid.Grid.prototype.loadMessage = "Bitte; warten..."; +} \ No newline at end of file diff --git a/admin/lib/js/grid.locale-ru.js b/admin/lib/js/grid.locale-ru.js new file mode 100644 index 0000000..3d3fdf0 --- /dev/null +++ b/admin/lib/js/grid.locale-ru.js @@ -0,0 +1,11 @@ +function translate_ru() { + window.jsGrid.Grid.prototype.noDataContent = "Не найдено"; + window.jsGrid.Grid.prototype.deleteConfirm = "Вы уверены?"; + window.jsGrid.Grid.prototype.pagerFormat = "Страницы = {first} {prev} {pages} {next} {last}    {pageIndex} of {pageCount}"; + window.jsGrid.Grid.prototype.pagePrevText = "Пред."; + window.jsGrid.Grid.prototype.pageNextText = "След"; + window.jsGrid.Grid.prototype.pageFirstText = "Первая"; + window.jsGrid.Grid.prototype.pageLastText = "Последняя"; + window.jsGrid.Grid.prototype.loadMessage = "Подождите..."; +} + diff --git a/admin/lib/js/jsgrid.js b/admin/lib/js/jsgrid.js new file mode 100644 index 0000000..c0208bd --- /dev/null +++ b/admin/lib/js/jsgrid.js @@ -0,0 +1,1954 @@ +/* + * jsGrid v1.0.1 (http://js-grid.com) + * (c) 2015 Artem Tabalin + * Licensed under MIT (https://github.com/tabalinas/jsgrid/blob/master/LICENSE) + */ + +(function(window, $, undefined) { + + var JSGRID = "JSGrid", + JSGRID_DATA_KEY = JSGRID, + JSGRID_ROW_DATA_KEY = "JSGridItem", + JSGRID_EDIT_ROW_DATA_KEY = "JSGridEditRow", + + SORT_ORDER_ASC = "asc", + SORT_ORDER_DESC = "desc", + + FIRST_PAGE_PLACEHOLDER = "{first}", + PAGES_PLACEHOLDER = "{pages}", + PREV_PAGE_PLACEHOLDER = "{prev}", + NEXT_PAGE_PLACEHOLDER = "{next}", + LAST_PAGE_PLACEHOLDER = "{last}", + PAGE_INDEX_PLACEHOLDER = "{pageIndex}", + PAGE_COUNT_PLACEHOLDER = "{pageCount}", + + EMPTY_HREF = "javascript:void(0);"; + + var getOrApply = function(value, context) { + if($.isFunction(value)) { + return value.apply(context, $.makeArray(arguments).slice(2)); + } + return value; + }; + + var defaultController = { + loadData: $.noop, + insertItem: $.noop, + updateItem: $.noop, + deleteItem: $.noop + }; + + + function Grid(element, config) { + var $element = $(element); + + $element.data(JSGRID_DATA_KEY, this); + + this._container = $element; + + this.data = []; + this.fields = []; + + this._editingRow = null; + this._sortField = null; + this._sortOrder = SORT_ORDER_ASC; + this._firstDisplayingPage = 1; + + this._init(config); + this.render(); + } + + Grid.prototype = { + width: "auto", + height: "auto", + updateOnResize: true, + + rowClass: $.noop, + rowRenderer: null, + + rowClick: function(args) { + if(this.editing) { + this.editItem($(args.event.target).closest("tr")); + } + }, + + noDataContent: "Not found", + noDataRowClass: "jsgrid-nodata-row", + + heading: true, + headerRowRenderer: null, + headerRowClass: "jsgrid-header-row", + + filtering: false, + filterRowRenderer: null, + filterRowClass: "jsgrid-filter-row", + + inserting: false, + insertRowRenderer: null, + insertRowClass: "jsgrid-insert-row", + + editing: false, + editRowRenderer: null, + editRowClass: "jsgrid-edit-row", + + confirmDeleting: true, + deleteConfirm: "Are you sure?", + + selecting: true, + selectedRowClass: "jsgrid-selected-row", + oddRowClass: "jsgrid-row", + evenRowClass: "jsgrid-alt-row", + + sorting: false, + sortableClass: "jsgrid-header-sortable", + sortAscClass: "jsgrid-header-sort jsgrid-header-sort-asc", + sortDescClass: "jsgrid-header-sort jsgrid-header-sort-desc", + + paging: false, + pagerContainer: null, + pageIndex: 1, + pageSize: 20, + pageButtonCount: 15, + pagerFormat: "Pages: {first} {prev} {pages} {next} {last}    {pageIndex} of {pageCount}", + pagePrevText: "Prev", + pageNextText: "Next", + pageFirstText: "First", + pageLastText: "Last", + pageNavigatorNextText: "...", + pageNavigatorPrevText: "...", + pagerContainerClass: "jsgrid-pager-container", + pagerClass: "jsgrid-pager", + pagerNavButtonClass: "jsgrid-pager-nav-button", + pageClass: "jsgrid-pager-page", + currentPageClass: "jsgrid-pager-current-page", + + pageLoading: false, + + autoload: false, + controller: defaultController, + + loadIndication: true, + loadIndicationDelay: 500, + loadMessage: "Please, wait...", + loadShading: true, + + onRefreshing: $.noop, + onRefreshed: $.noop, + onItemDeleting: $.noop, + onItemDeleted: $.noop, + onItemInserting: $.noop, + onItemInserted: $.noop, + onItemUpdating: $.noop, + onItemUpdated: $.noop, + onDataLoading: $.noop, + onDataLoaded: $.noop, + onOptionChanging: $.noop, + onOptionChanged: $.noop, + onError: $.noop, + + containerClass: "jsgrid", + tableClass: "jsgrid-table", + gridHeaderClass: "jsgrid-grid-header", + gridBodyClass: "jsgrid-grid-body", + + _init: function(config) { + $.extend(this, config); + this._initLoadStrategy(); + this._initController(); + this._initFields(); + this._attachWindowLoadResize(); + this._attachWindowResizeCallback(); + }, + + loadStrategy: function() { + return this.pageLoading + ? new jsGrid.loadStrategies.PageLoadingStrategy(this) + : new jsGrid.loadStrategies.DirectLoadingStrategy(this); + }, + + _initLoadStrategy: function() { + this._loadStrategy = getOrApply(this.loadStrategy, this); + }, + + _initController: function() { + this._controller = $.extend({}, defaultController, getOrApply(this.controller, this)); + }, + + loadIndicator: function(config) { + return new jsGrid.LoadIndicator(config); + }, + + _initFields: function() { + var self = this; + self.fields = $.map(self.fields, function(field) { + if($.isPlainObject(field)) { + var fieldConstructor = (field.type && jsGrid.fields[field.type]) || jsGrid.Field; + field = new fieldConstructor(field); + } + field._grid = self; + return field; + }); + }, + + _attachWindowLoadResize: function() { + $(window).on("load", $.proxy(this._refreshSize, this)); + }, + + _attachWindowResizeCallback: function() { + if(this.updateOnResize) { + $(window).on("resize", $.proxy(this._refreshSize, this)); + } + }, + + _detachWindowResizeCallback: function() { + $(window).off("resize", this._refreshSize); + }, + + option: function(key, value) { + var optionChangingEventArgs, + optionChangedEventArgs; + + if(arguments.length === 1) { + return this[key]; + } + + optionChangingEventArgs = { + option: key, + oldValue: this[key], + newValue: value + }; + this._callEventHandler(this.onOptionChanging, optionChangingEventArgs); + + this._handleOptionChange(optionChangingEventArgs.option, optionChangingEventArgs.newValue); + + optionChangedEventArgs = { + option: optionChangingEventArgs.option, + value: optionChangingEventArgs.newValue + }; + this._callEventHandler(this.onOptionChanged, optionChangedEventArgs); + }, + + _handleOptionChange: function(name, value) { + this[name] = value; + + switch(name) { + case "width": + case "height": + this._refreshSize(); + break; + case "rowClass": + case "rowRenderer": + case "rowClick": + case "noDataText": + case "noDataRowClass": + case "noDataContent": + case "selecting": + case "selectedRowClass": + case "oddRowClass": + case "evenRowClass": + this._refreshContent(); + break; + case "pageButtonCount": + case "pagerFormat": + case "pagePrevText": + case "pageNextText": + case "pageFirstText": + case "pageLastText": + case "pageNavigatorNextText": + case "pageNavigatorPrevText": + case "pagerClass": + case "pagerNavButtonClass": + case "pageClass": + case "currentPageClass": + this._refreshPager(); + break; + case "fields": + this._initFields(); + this.render(); + break; + case "data": + case "editing": + case "heading": + case "filtering": + case "inserting": + case "paging": + this.refresh(); + break; + case "pageLoading": + this._initLoadStrategy(); + this.search(); + break; + case "pageIndex": + this.openPage(value); + break; + case "pageSize": + this.refresh(); + this.search(); + break; + case "editRowRenderer": + case "editRowClass": + this.cancelEdit(); + break; + default: + this.render(); + break; + } + }, + + destroy: function() { + this._detachWindowResizeCallback(); + this._clear(); + this._container.removeData(JSGRID_DATA_KEY); + }, + + render: function() { + this._clear(); + + this._container.addClass(this.containerClass) + .css("position", "relative") + .append(this._createHeader()) + .append(this._createBody()); + + this._pagerContainer = this._createPagerContainer(); + this._loadIndicator = this._createLoadIndicator(); + + this.refresh(); + + return this.autoload ? this.loadData() : $.Deferred().resolve().promise(); + }, + + _createLoadIndicator: function() { + return getOrApply(this.loadIndicator, this, { + message: this.loadMessage, + shading: this.loadShading, + container: this._container + }); + }, + + _clear: function() { + this.cancelEdit(); + + clearTimeout(this._loadingTimer); + + this._pagerContainer && this._pagerContainer.empty(); + + this._container.empty() + .css({ position: "", width: "", height: "" }); + }, + + _createHeader: function() { + var $headerRow = this._headerRow = this._createHeaderRow(), + $filterRow = this._filterRow = this._createFilterRow(), + $insertRow = this._insertRow = this._createInsertRow(); + + var $headerGrid = this._headerGrid = $("").addClass(this.tableClass) + .append($headerRow) + .append($filterRow) + .append($insertRow); + + var $header = this._header = $("
").addClass(this.gridHeaderClass) + .append($headerGrid); + + return $header; + }, + + _createBody: function() { + var $content = this._content = $("
"); + + var $bodyGrid = this._bodyGrid = $("
").addClass(this.tableClass) + .append($content); + + var $body = this._body = $("
").addClass(this.gridBodyClass) + .append($bodyGrid); + + return $body; + }, + + _createPagerContainer: function() { + var pagerContainer = this.pagerContainer || $("
").appendTo(this._container); + return $(pagerContainer).addClass(this.pagerContainerClass); + }, + + _eachField: function(callBack) { + var self = this; + $.each(this.fields, function(index, field) { + return callBack.call(self, field, index); + }); + }, + + _createHeaderRow: function() { + if($.isFunction(this.headerRowRenderer)) { + return $(this.headerRowRenderer()); + } + + var $result = $("
").addClass(this.headerRowClass); + + this._eachField(function(field, index) { + var $th = $("").addClass(this.filterRowClass); + + this._eachField(function(field) { + $("").addClass(this.insertRowClass); + + this._eachField(function(field) { + $("").addClass(this.noDataRowClass) + .append($(""); + this._renderCells($result, item); + } + + $result.addClass(this._getRowClasses(item, itemIndex)) + .data(JSGRID_ROW_DATA_KEY, item) + .on("click", $.proxy(function(e) { + this.rowClick({ + item: item, + itemIndex: itemIndex, + event: e + }); + }, this)); + + if(this.selecting) { + this._attachRowHover($result); + } + + return $result; + }, + + _getRowClasses: function(item, itemIndex) { + var classes = []; + classes.push(((itemIndex + 1) % 2) ? this.oddRowClass : this.evenRowClass); + classes.push(getOrApply(this.rowClass, this, item, itemIndex)); + return classes.join(" "); + }, + + _attachRowHover: function($row) { + var selectedRowClass = this.selectedRowClass; + $row.hover(function() { + $(this).addClass(selectedRowClass); + }, + function() { + $(this).removeClass(selectedRowClass); + } + ); + }, + + _renderCells: function($row, item) { + this._eachField(function(field) { + $row.append(this._createCell(item, field)); + }); + return this; + }, + + _createCell: function(item, field) { + var $result; + var fieldValue = item[field.name]; + + if($.isFunction(field.cellRenderer)) { + $result = $(field.cellRenderer(fieldValue, item)); + } else { + $result = $("").addClass(this.editRowClass); + + this._eachField(function(field) { + $("
").addClass(field.css) + .appendTo($result) + .append(field.headerTemplate ? field.headerTemplate() : "") + .css("width", field.width); + + if(this.sorting && field.sorting) { + $th.addClass(this.sortableClass) + .on("click", $.proxy(function() { + this.sort(index); + }, this)); + } + }); + + return $result; + }, + + _createFilterRow: function() { + if($.isFunction(this.filterRowRenderer)) { + return $(this.filterRowRenderer()); + } + + var $result = $("
").addClass(field.css) + .appendTo($result) + .append(field.filterTemplate ? field.filterTemplate() : "") + .width(field.width); + }); + + return $result; + }, + + _createInsertRow: function() { + if($.isFunction(this.insertRowRenderer)) { + return $(this.insertRowRenderer()); + } + + var $result = $("
").addClass(field.css) + .appendTo($result) + .append(field.insertTemplate ? field.insertTemplate() : "") + .width(field.width); + }); + + return $result; + }, + + _callEventHandler: function(handler, eventParams) { + return handler.call(this, $.extend(eventParams, { + grid: this + })); + }, + + reset: function() { + this._resetSorting(); + this._resetPager(); + this.refresh(); + }, + + _resetPager: function() { + this._firstDisplayingPage = 1; + this._setPage(1); + }, + + _resetSorting: function() { + this._sortField = null; + this._sortOrder = SORT_ORDER_ASC; + this._clearSortingCss(); + }, + + refresh: function() { + this._callEventHandler(this.onRefreshing); + + this.cancelEdit(); + + this._refreshHeading(); + this._refreshFiltering(); + this._refreshInserting(); + this._refreshContent(); + this._refreshPager(); + this._refreshSize(); + + this._callEventHandler(this.onRefreshed); + }, + + _refreshHeading: function() { + this._headerRow.toggle(this.heading); + }, + + _refreshFiltering: function() { + this._filterRow.toggle(this.filtering); + }, + + _refreshInserting: function() { + this._insertRow.toggle(this.inserting); + }, + + _refreshContent: function() { + var $content = this._content; + $content.empty(); + + if(!this.data.length) { + $content.append(this._createNoDataRow()); + return this; + } + + var indexFrom = this._loadStrategy.firstDisplayIndex(); + var indexTo = this._loadStrategy.lastDisplayIndex(); + + for(var itemIndex = indexFrom; itemIndex < indexTo; itemIndex++) { + var item = this.data[itemIndex]; + $content.append(this._createRow(item, itemIndex)); + } + }, + + _createNoDataRow: function() { + var noDataContent = getOrApply(this.noDataContent, this); + return $("
").attr("colspan", this.fields.length).append(noDataContent)); + }, + + _createNoDataContent: function () { + return $.isFunction(this.noDataRenderer) + ? this.noDataRenderer() + : this.noDataText; + }, + + _createRow: function(item, itemIndex) { + var $result; + + if($.isFunction(this.rowRenderer)) { + $result = $(this.rowRenderer(item, itemIndex)); + } else { + $result = $("
").append(field.itemTemplate ? field.itemTemplate(fieldValue, item) : fieldValue); + } + + $result.addClass(field.css) + .width(field.width); + + field.align && $result.addClass("jsgrid-align-" + field.align); + + return $result; + }, + + sort: function(field, order) { + if($.isPlainObject(field)) { + order = field.order; + field = field.field; + } + + this._clearSortingCss(); + this._setSortingParams(field, order); + this._setSortingCss(); + return this._loadStrategy.sort(); + }, + + _clearSortingCss: function() { + this._headerRow.find("th") + .removeClass(this.sortAscClass) + .removeClass(this.sortDescClass); + }, + + _setSortingParams: function(field, order) { + field = this._normalizeSortingField(field); + order = order || ((this._sortField === field) ? this._reversedSortOrder(this._sortOrder) : SORT_ORDER_ASC); + + this._sortField = field; + this._sortOrder = order; + }, + + _normalizeSortingField: function(field) { + if($.isNumeric(field)) { + return this.fields[field]; + } + + if(typeof field === "string") { + return $.grep(this.fields, function (f) { + return f.name === field; + })[0]; + } + + return field; + }, + + _reversedSortOrder: function(order) { + return (order === SORT_ORDER_ASC ? SORT_ORDER_DESC : SORT_ORDER_ASC); + }, + + _setSortingCss: function() { + var fieldIndex = $.inArray(this._sortField, this.fields); + + this._headerRow.find("th").eq(fieldIndex) + .addClass(this._sortOrder === SORT_ORDER_ASC ? this.sortAscClass : this.sortDescClass); + }, + + _sortData: function() { + var sortFactor = this._sortFactor(), + sortField = this._sortField; + + if(sortField) { + this.data.sort(function(item1, item2) { + return sortFactor * sortField.sortingFunc(item1[sortField.name], item2[sortField.name]); + }); + } + }, + + _sortFactor: function() { + return this._sortOrder === SORT_ORDER_ASC ? 1 : -1; + }, + + _itemsCount: function() { + return this._loadStrategy.itemsCount(); + }, + + _pagesCount: function() { + var itemsCount = this._itemsCount(), + pageSize = this.pageSize; + return Math.floor(itemsCount / pageSize) + (itemsCount % pageSize ? 1 : 0); + }, + + _refreshPager: function() { + var $pagerContainer = this._pagerContainer; + $pagerContainer.empty(); + + if(this.paging && this._pagesCount() > 1) { + $pagerContainer.show() + .append(this._createPager()); + } else { + $pagerContainer.hide(); + } + }, + + _createPager: function() { + var pageIndex = this.pageIndex, + pageCount = this._pagesCount(), + pagerParts = this.pagerFormat.split(" "); + + pagerParts = $.map(pagerParts, $.proxy(function(pagerPart) { + var result = pagerPart; + + if(pagerPart === PAGES_PLACEHOLDER) { + result = this._createPages(); + } else if(pagerPart === FIRST_PAGE_PLACEHOLDER) { + result = pageIndex > 1 ? this._createPagerNavButton(this.pageFirstText, 1) : ""; + } else if(pagerPart === PREV_PAGE_PLACEHOLDER) { + result = pageIndex > 1 ? this._createPagerNavButton(this.pagePrevText, pageIndex - 1) : ""; + } else if(pagerPart === NEXT_PAGE_PLACEHOLDER) { + result = pageIndex < pageCount ? this._createPagerNavButton(this.pageNextText, pageIndex + 1) : ""; + } else if(pagerPart === LAST_PAGE_PLACEHOLDER) { + result = pageIndex < pageCount ? this._createPagerNavButton(this.pageLastText, pageCount) : ""; + } else if(pagerPart === PAGE_INDEX_PLACEHOLDER) { + result = pageIndex; + } else if(pagerPart === PAGE_COUNT_PLACEHOLDER) { + result = pageCount; + } + + return $.isArray(result) ? result.concat([" "]) : [result, " "]; + }, this)); + + var $pager = $("
").addClass(this.pagerClass) + .append(pagerParts); + + return $pager; + }, + + _createPages: function() { + var pageCount = this._pagesCount(), + pageButtonCount = this.pageButtonCount, + firstDisplayingPage = this._firstDisplayingPage, + pages = [], + pageNumber; + + if(firstDisplayingPage > 1) { + pages.push(this._createPagerPageNavButton(this.pageNavigatorPrevText, this.showPrevPages)); + } + + for(var i = 0, pageNumber = firstDisplayingPage; i < pageButtonCount && pageNumber <= pageCount; i++, pageNumber++) { + pages.push(pageNumber === this.pageIndex + ? this._createPagerCurrentPage() + : this._createPagerPage(pageNumber)); + } + + if((firstDisplayingPage + pageButtonCount - 1) < pageCount) { + pages.push(this._createPagerPageNavButton(this.pageNavigatorNextText, this.showNextPages)); + } + + return pages; + }, + + _createPagerNavButton: function(text, pageIndex) { + return this._createPagerButton(text, this.pagerNavButtonClass, function() { + this.openPage(pageIndex); + }); + }, + + _createPagerPageNavButton: function(text, handler) { + return this._createPagerButton(text, this.pagerNavButtonClass, handler); + }, + + _createPagerPage: function(pageIndex) { + return this._createPagerButton(pageIndex, this.pageClass, function() { + this.openPage(pageIndex); + }); + }, + + _createPagerButton: function(text, css, handler) { + var $link = $("").attr("href", EMPTY_HREF) + .html(text) + .on("click", $.proxy(handler, this)); + + return $("").addClass(css).append($link); + }, + + _createPagerCurrentPage: function() { + return $("") + .addClass(this.pageClass) + .addClass(this.currentPageClass) + .text(this.pageIndex); + }, + + _refreshSize: function() { + this._refreshHeight(); + this._refreshWidth(); + }, + + _refreshWidth: function() { + var $headerGrid = this._headerGrid, + $bodyGrid = this._bodyGrid, + width = this.width, + scrollBarWidth = this._scrollBarWidth(), + gridWidth; + + if(width === "auto") { + $headerGrid.width("auto"); + gridWidth = $headerGrid.outerWidth(); + width = gridWidth + scrollBarWidth; + } + + $headerGrid.width(""); + $bodyGrid.width(""); + this._header.css("padding-right", scrollBarWidth); + this._container.width(width); + gridWidth = $headerGrid.outerWidth(); + $bodyGrid.width(gridWidth); + }, + + _scrollBarWidth: (function() { + var result; + + return function() { + if(result === undefined) { + var $ghostContainer = $("
"); + var $ghostContent = $("
"); + $ghostContainer.append($ghostContent).appendTo("body"); + var width = $ghostContent.innerWidth(); + $ghostContainer.css("overflow-y", "auto"); + var widthExcludingScrollBar = $ghostContent.innerWidth(); + $ghostContainer.remove(); + result = width - widthExcludingScrollBar; + } + return result; + }; + })(), + + _refreshHeight: function() { + var container = this._container, + pagerContainer = this._pagerContainer, + height = this.height, + nonBodyHeight; + + container.height(height); + + if(height !== "auto") { + height = container.height(); + + nonBodyHeight = this._header.outerHeight(true); + if(pagerContainer.parents(container).length) { + nonBodyHeight += pagerContainer.outerHeight(true); + } + + this._body.outerHeight(height - nonBodyHeight); + } + }, + + showPrevPages: function() { + var firstDisplayingPage = this._firstDisplayingPage, + pageButtonCount = this.pageButtonCount; + + this._firstDisplayingPage = (firstDisplayingPage > pageButtonCount) ? firstDisplayingPage - pageButtonCount : 1; + + this._refreshPager(); + }, + + showNextPages: function() { + var firstDisplayingPage = this._firstDisplayingPage, + pageButtonCount = this.pageButtonCount, + pageCount = this._pagesCount(); + + this._firstDisplayingPage = (firstDisplayingPage + 2 * pageButtonCount > pageCount) + ? pageCount - pageButtonCount + 1 + : firstDisplayingPage + pageButtonCount; + + this._refreshPager(); + }, + + openPage: function(pageIndex) { + if(pageIndex < 1 || pageIndex > this._pagesCount()) + return; + + this._setPage(pageIndex); + this._loadStrategy.openPage(pageIndex); + }, + + _setPage: function(pageIndex) { + var firstDisplayingPage = this._firstDisplayingPage, + pageButtonCount = this.pageButtonCount; + + this.pageIndex = pageIndex; + + if(pageIndex < firstDisplayingPage) { + this._firstDisplayingPage = pageIndex; + } + + if(pageIndex > firstDisplayingPage + pageButtonCount - 1) { + this._firstDisplayingPage = pageIndex - pageButtonCount + 1; + } + }, + + _controllerCall: function(method, param, doneCallback) { + this._showLoading(); + + var controller = this._controller; + if(!controller || !controller[method]) { + throw new Error("controller has no method '" + method + "'"); + } + + return $.when(controller[method](param)) + .done($.proxy(doneCallback, this)) + .fail($.proxy(this._errorHandler, this)) + .always($.proxy(this._hideLoading, this)); + }, + + _errorHandler: function() { + this._callEventHandler(this.onError, { + args: $.makeArray(arguments) + }); + }, + + _showLoading: function() { + clearTimeout(this._loadingTimer); + + this._loadingTimer = setTimeout($.proxy(function() { + this._loadIndicator.show(); + }, this), this.loadIndicationDelay); + }, + + _hideLoading: function() { + clearTimeout(this._loadingTimer); + this._loadIndicator.hide(); + }, + + search: function(filter) { + this._resetSorting(); + this._resetPager(); + return this.loadData(filter); + }, + + loadData: function(filter) { + filter = filter || (this.filtering ? this._getFilter() : {}); + + $.extend(filter, this._loadStrategy.loadParams(), this._sortingParams()); + + this._callEventHandler(this.onDataLoading, { + filter: filter + }); + + return this._controllerCall("loadData", filter, function(loadedData) { + this._loadStrategy.finishLoad(loadedData); + + this._callEventHandler(this.onDataLoaded, { + data: loadedData + }); + }); + }, + + _getFilter: function() { + var result = {}; + this._eachField(function(field) { + if(field.filtering) { + result[field.name] = field.filterValue(); + } + }); + return result; + }, + + _sortingParams: function() { + if(this.sorting && this._sortField) { + return { + sortField: this._sortField.name, + sortOrder: this._sortOrder + }; + } + return {}; + }, + + clearFilter: function() { + var $filterRow = this._createFilterRow(); + this._filterRow.replaceWith($filterRow); + this._filterRow = $filterRow; + return this.search(); + }, + + insertItem: function(item) { + var insertingItem = item || this._getInsertItem(); + + this._callEventHandler(this.onItemInserting, { + item: insertingItem + }); + + return this._controllerCall("insertItem", insertingItem, function(insertedItem) { + insertedItem = insertedItem || insertingItem; + this._loadStrategy.finishInsert(insertedItem); + + this._callEventHandler(this.onItemInserted, { + item: insertedItem + }); + }); + }, + + _getInsertItem: function() { + var result = {}; + this._eachField(function(field) { + if(field.inserting) { + result[field.name] = field.insertValue(); + } + }); + return result; + }, + + clearInsert: function() { + var insertRow = this._createInsertRow(); + this._insertRow.replaceWith(insertRow); + this._insertRow = insertRow; + this.refresh(); + }, + + editItem: function(item) { + var $row = this._rowByItem(item); + if($row.length) { + this._editRow($row); + } + }, + + _rowByItem: function(item) { + if(item.jquery || item.nodeType) + return $(item); + + return this._content.find("tr").filter(function() { + return $.data(this, JSGRID_ROW_DATA_KEY) === item; + }); + }, + + _editRow: function($row) { + if(!this.editing) + return; + + if(this._editingRow) { + this.cancelEdit(); + } + + var item = $row.data(JSGRID_ROW_DATA_KEY), + $editRow = this._createEditRow(item); + + this._editingRow = $row; + $row.hide(); + $editRow.insertAfter($row); + $row.data(JSGRID_EDIT_ROW_DATA_KEY, $editRow); + }, + + _createEditRow: function(item) { + if($.isFunction(this.editRowRenderer)) { + return $(this.editRowRenderer(item, this._itemIndex(item))); + } + + var $result = $("
").addClass(field.css) + .appendTo($result) + .append(field.editTemplate ? field.editTemplate(item[field.name], item) : "") + .width(field.width || "auto"); + }); + + return $result; + }, + + updateItem: function(item, editedItem) { + if(arguments.length === 1) { + editedItem = item; + } + + var $row = item ? this._rowByItem(item) : this._editingRow; + editedItem = editedItem || this._getEditedItem(); + + return this._updateRow($row, editedItem); + }, + + _updateRow: function($updatingRow, editedItem) { + var updatingItem = $updatingRow.data(JSGRID_ROW_DATA_KEY), + updatingItemIndex = this._itemIndex(updatingItem); + + $.extend(updatingItem, editedItem); + + this._callEventHandler(this.onItemUpdating, { + row: $updatingRow, + item: updatingItem, + itemIndex: updatingItemIndex + }); + + return this._controllerCall("updateItem", updatingItem, function(updatedItem) { + updatedItem = updatedItem || updatingItem; + this._finishUpdate($updatingRow, updatedItem, updatingItemIndex); + + this._callEventHandler(this.onItemUpdated, { + row: $updatingRow, + item: updatedItem, + itemIndex: updatingItemIndex + }); + }); + }, + + _itemIndex: function(item) { + return $.inArray(item, this.data); + }, + + _finishUpdate: function($updatedRow, updatedItem, updatedItemIndex) { + this.cancelEdit(); + this.data[updatedItemIndex] = updatedItem; + $updatedRow.replaceWith(this._createRow(updatedItem, updatedItemIndex)); + }, + + _getEditedItem: function() { + var result = {}; + this._eachField(function(field) { + if(field.editing) { + result[field.name] = field.editValue(); + } + }); + return result; + }, + + cancelEdit: function() { + if(!this._editingRow) { + return; + } + + var $row = this._editingRow, + $editRow = $row.data(JSGRID_EDIT_ROW_DATA_KEY); + + $editRow.remove(); + $row.show(); + this._editingRow = null; + }, + + deleteItem: function(item) { + var $row = this._rowByItem(item); + + if(!$row.length) + return; + + var that = this; + if (typeof this.deleteConfirm === 'function') { + this.deleteConfirm(this, $row.data(JSGRID_ROW_DATA_KEY), function (result) { + if (result) { + that._deleteRow($row); + } + }); + } else { + if(this.confirmDeleting && !window.confirm(getOrApply(this.deleteConfirm, this, $row.data(JSGRID_ROW_DATA_KEY)))) + return; + + return this._deleteRow($row); + } + }, + + _deleteRow: function($row) { + var deletingItem = $row.data(JSGRID_ROW_DATA_KEY), + deletingItemIndex = this._itemIndex(deletingItem); + + this._callEventHandler(this.onItemDeleting, { + row: $row, + item: deletingItem, + itemIndex: deletingItemIndex + }); + + return this._controllerCall("deleteItem", deletingItem, function() { + this._loadStrategy.finishDelete(deletingItem, deletingItemIndex); + + this._callEventHandler(this.onItemDeleted, { + row: $row, + item: deletingItem, + itemIndex: deletingItemIndex + }); + }); + } + }; + + $.fn.jsGrid = function(config) { + var args = $.makeArray(arguments), + methodArgs = args.slice(1), + result = this; + + this.each(function() { + var $element = $(this), + instance = $element.data(JSGRID_DATA_KEY), + methodResult; + + if(instance) { + if(typeof config === "string") { + methodResult = instance[config].apply(instance, methodArgs); + if(methodResult !== undefined && methodResult !== instance) { + result = methodResult; + return false; + } + } else { + instance._detachWindowResizeCallback(); + instance._init(config); + instance.render(); + } + } else { + new Grid($element, config); + } + }); + + return result; + }; + + window.jsGrid = { + Grid: Grid, + fields: [] + }; + +}(window, jQuery)); +(function(jsGrid, $, undefined) { + + function LoadIndicator(config) { + this._init(config); + } + + LoadIndicator.prototype = { + + container: "body", + message: "Loading...", + shading: true, + + zIndex: 1000, + shaderClass: "jsgrid-load-shader", + loadPanelClass: "jsgrid-load-panel", + + _init: function(config) { + $.extend(true, this, config); + + this._initContainer(); + this._initShader(); + this._initLoadPanel(); + }, + + _initContainer: function() { + this._container = $(this.container); + }, + + _initShader: function() { + if(!this.shading) + return; + + this._shader = $("
").addClass(this.shaderClass) + .hide() + .css({ + position: "absolute", + top: 0, + right: 0, + bottom: 0, + left: 0, + zIndex: this.zIndex + }) + .appendTo(this._container); + }, + + _initLoadPanel: function() { + this._loadPanel = $("
").addClass(this.loadPanelClass) + .text(this.message) + .hide() + .css({ + position: "absolute", + top: "50%", + left: "50%", + zIndex: this.zIndex + }) + .appendTo(this._container); + }, + + show: function() { + var $loadPanel = this._loadPanel.show(); + + var actualWidth = $loadPanel.outerWidth(); + var actualHeight = $loadPanel.outerHeight(); + + $loadPanel.css({ + marginTop: -actualHeight / 2, + marginLeft: -actualWidth / 2 + }); + + this._shader.show(); + }, + + hide: function() { + this._loadPanel.hide(); + this._shader.hide(); + } + + }; + + jsGrid.LoadIndicator = LoadIndicator; + +}(jsGrid, jQuery)); +(function(jsGrid, $, undefined) { + + function DirectLoadingStrategy(grid) { + this._grid = grid; + } + + DirectLoadingStrategy.prototype = { + + firstDisplayIndex: function() { + var grid = this._grid; + return grid.option("paging") ? (grid.option("pageIndex") - 1) * grid.option("pageSize") : 0; + }, + + lastDisplayIndex: function() { + var grid = this._grid; + var itemsCount = grid.option("data").length; + + return grid.option("paging") + ? Math.min(grid.option("pageIndex") * grid.option("pageSize"), itemsCount) + : itemsCount; + }, + + itemsCount: function() { + return this._grid.option("data").length; + }, + + openPage: function(index) { + this._grid.refresh(); + }, + + loadParams: function() { + return {}; + }, + + sort: function() { + this._grid._sortData(); + this._grid.refresh(); + return $.Deferred().resolve().promise(); + }, + + finishLoad: function(loadedData) { + this._grid.option("data", loadedData); + }, + + finishInsert: function(insertedItem) { + var grid = this._grid; + grid.option("data").push(insertedItem); + grid.refresh(); + }, + + finishDelete: function(deletedItem, deletedItemIndex) { + var grid = this._grid; + grid.option("data").splice(deletedItemIndex, 1); + grid.reset(); + } + }; + + + function PageLoadingStrategy(grid) { + this._grid = grid; + this._itemsCount = 0; + } + + PageLoadingStrategy.prototype = { + firstDisplayIndex: function() { + return 0; + }, + + lastDisplayIndex: function() { + return this._grid.option("data").length; + }, + + itemsCount: function() { + return this._itemsCount; + }, + + openPage: function(index) { + this._grid.loadData(); + }, + + loadParams: function() { + var grid = this._grid; + return { + pageIndex: grid.option("pageIndex"), + pageSize: grid.option("pageSize") + }; + }, + + sort: function() { + return this._grid.loadData(); + }, + + finishLoad: function(loadedData) { + this._itemsCount = loadedData.itemsCount; + this._grid.option("data", loadedData.data); + }, + + finishInsert: function(insertedItem) { + this._grid.search(); + }, + + finishDelete: function(deletedItem, deletedItemIndex) { + this._grid.search(); + } + }; + + jsGrid.loadStrategies = { + DirectLoadingStrategy: DirectLoadingStrategy, + PageLoadingStrategy: PageLoadingStrategy + }; + +}(jsGrid, jQuery)); +(function(jsGrid, $, undefined) { + + var sortStrategies = { + string: function(str1, str2) { + return str1.localeCompare(str2); + }, + + number: function(n1, n2) { + return n1 - n2; + }, + + date: function(dt1, dt2) { + return dt1 - dt2; + }, + + numberAsString: function(n1, n2) { + return parseFloat(n1) - parseFloat(n2); + } + }; + + jsGrid.sortStrategies = sortStrategies; + +}(jsGrid, jQuery)); +(function(jsGrid, $, undefined) { + + function Field(config) { + $.extend(true, this, config); + this.sortingFunc = this._getSortingFunc(); + } + + Field.prototype = { + name: "", + title: "", + css: "", + align: "", + width: 100, + + filtering: true, + inserting: true, + editing: true, + sorting: true, + sorter: "string", // name of SortStrategy or function to compare elements + + headerTemplate: function() { + return this.title || this.name; + }, + + itemTemplate: function(value, item) { + return value; + }, + + filterTemplate: function() { + return ""; + }, + + insertTemplate: function() { + return ""; + }, + + editTemplate: function(value, item) { + this._value = value; + return this.itemTemplate(value, item); + }, + + filterValue: function() { + return ""; + }, + + insertValue: function() { + return ""; + }, + + editValue: function() { + return this._value; + }, + + _getSortingFunc: function() { + var sorter = this.sorter; + + if($.isFunction(sorter)) { + return sorter; + } + + if(typeof sorter === "string") { + return jsGrid.sortStrategies[sorter]; + } + + throw Error("Wrong sorter for the field \"" + this.name + "\"!"); + } + }; + + jsGrid.Field = Field; + +}(jsGrid, jQuery)); +(function(jsGrid, $, undefined) { + + var Field = jsGrid.Field; + + function TextField(config) { + Field.call(this, config); + } + + TextField.prototype = new Field({ + + autosearch: true, + + filterTemplate: function() { + var grid = this._grid, + $result = this.filterControl = this._createTextBox(); + + if(this.autosearch) { + $result.on("keypress", function(e) { + if(e.which === 13) { + grid.search(); + e.preventDefault(); + } + }); + } + + return $result; + }, + + insertTemplate: function() { + var $result = this.insertControl = this._createTextBox(); + return $result; + }, + + editTemplate: function(value) { + var $result = this.editControl = this._createTextBox(); + $result.val(value); + return $result; + }, + + filterValue: function() { + return this.filterControl.val(); + }, + + insertValue: function() { + return this.insertControl.val(); + }, + + editValue: function() { + return this.editControl.val(); + }, + + _createTextBox: function() { + return $("").attr("type", "text"); + } + }); + + jsGrid.fields.text = jsGrid.TextField = TextField; + +}(jsGrid, jQuery)); +(function(jsGrid, $, undefined) { + + var TextField = jsGrid.TextField; + + function NumberField(config) { + TextField.call(this, config); + } + + NumberField.prototype = new TextField({ + + sorter: "number", + align: "right", + + filterValue: function() { + return parseInt(this.filterControl.val() || 0, 10); + }, + + insertValue: function() { + return parseInt(this.insertControl.val() || 0, 10); + }, + + editValue: function() { + return parseInt(this.editControl.val() || 0, 10); + }, + + _createTextBox: function() { + return $("").attr("type", "number"); + } + }); + + jsGrid.fields.number = jsGrid.NumberField = NumberField; + +}(jsGrid, jQuery)); +(function(jsGrid, $, undefined) { + + var TextField = jsGrid.TextField; + + function TextAreaField(config) { + TextField.call(this, config); + } + + TextAreaField.prototype = new TextField({ + + insertTemplate: function() { + var $result = this.insertControl = this._createTextArea(); + return $result; + }, + + editTemplate: function(value) { + var $result = this.editControl = this._createTextArea(); + $result.val(value); + return $result; + }, + + _createTextArea: function() { + return $("