Initial Commit
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
nbproject
|
9
.gitmodules
vendored
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
[submodule "protected/modules/feedback"]
|
||||||
|
path = protected/modules/feedback
|
||||||
|
url = git@git.humhub.org:/home/git/humhub-modules-feedback.git
|
||||||
|
[submodule "protected/modules/translation"]
|
||||||
|
path = protected/modules/translation
|
||||||
|
url = git@github.com:humhub/humhub-modules-translation.git
|
||||||
|
[submodule "themes/enoa"]
|
||||||
|
path = themes/enoa
|
||||||
|
url = git@git.humhub.org:/home/git/humhub-themes-enoa.git
|
9
.htaccess
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
RewriteEngine on
|
||||||
|
|
||||||
|
# if a directory or a file exists, use it directly
|
||||||
|
RewriteCond %{REQUEST_FILENAME} !-f
|
||||||
|
RewriteCond %{REQUEST_FILENAME} !-d
|
||||||
|
|
||||||
|
# otherwise forward it to index.php
|
||||||
|
RewriteRule . index.php
|
||||||
|
|
661
LICENSE
@ -1,661 +0,0 @@
|
|||||||
GNU AFFERO GENERAL PUBLIC LICENSE
|
|
||||||
Version 3, 19 November 2007
|
|
||||||
|
|
||||||
Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
|
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
|
||||||
of this license document, but changing it is not allowed.
|
|
||||||
|
|
||||||
Preamble
|
|
||||||
|
|
||||||
The GNU Affero General Public License is a free, copyleft license for
|
|
||||||
software and other kinds of works, specifically designed to ensure
|
|
||||||
cooperation with the community in the case of network server software.
|
|
||||||
|
|
||||||
The licenses for most software and other practical works are designed
|
|
||||||
to take away your freedom to share and change the works. By contrast,
|
|
||||||
our General Public Licenses are intended to guarantee your freedom to
|
|
||||||
share and change all versions of a program--to make sure it remains free
|
|
||||||
software for all its users.
|
|
||||||
|
|
||||||
When we speak of free software, we are referring to freedom, not
|
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
|
||||||
have the freedom to distribute copies of free software (and charge for
|
|
||||||
them if you wish), that you receive source code or can get it if you
|
|
||||||
want it, that you can change the software or use pieces of it in new
|
|
||||||
free programs, and that you know you can do these things.
|
|
||||||
|
|
||||||
Developers that use our General Public Licenses protect your rights
|
|
||||||
with two steps: (1) assert copyright on the software, and (2) offer
|
|
||||||
you this License which gives you legal permission to copy, distribute
|
|
||||||
and/or modify the software.
|
|
||||||
|
|
||||||
A secondary benefit of defending all users' freedom is that
|
|
||||||
improvements made in alternate versions of the program, if they
|
|
||||||
receive widespread use, become available for other developers to
|
|
||||||
incorporate. Many developers of free software are heartened and
|
|
||||||
encouraged by the resulting cooperation. However, in the case of
|
|
||||||
software used on network servers, this result may fail to come about.
|
|
||||||
The GNU General Public License permits making a modified version and
|
|
||||||
letting the public access it on a server without ever releasing its
|
|
||||||
source code to the public.
|
|
||||||
|
|
||||||
The GNU Affero General Public License is designed specifically to
|
|
||||||
ensure that, in such cases, the modified source code becomes available
|
|
||||||
to the community. It requires the operator of a network server to
|
|
||||||
provide the source code of the modified version running there to the
|
|
||||||
users of that server. Therefore, public use of a modified version, on
|
|
||||||
a publicly accessible server, gives the public access to the source
|
|
||||||
code of the modified version.
|
|
||||||
|
|
||||||
An older license, called the Affero General Public License and
|
|
||||||
published by Affero, was designed to accomplish similar goals. This is
|
|
||||||
a different license, not a version of the Affero GPL, but Affero has
|
|
||||||
released a new version of the Affero GPL which permits relicensing under
|
|
||||||
this license.
|
|
||||||
|
|
||||||
The precise terms and conditions for copying, distribution and
|
|
||||||
modification follow.
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
0. Definitions.
|
|
||||||
|
|
||||||
"This License" refers to version 3 of the GNU Affero General Public License.
|
|
||||||
|
|
||||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
|
||||||
works, such as semiconductor masks.
|
|
||||||
|
|
||||||
"The Program" refers to any copyrightable work licensed under this
|
|
||||||
License. Each licensee is addressed as "you". "Licensees" and
|
|
||||||
"recipients" may be individuals or organizations.
|
|
||||||
|
|
||||||
To "modify" a work means to copy from or adapt all or part of the work
|
|
||||||
in a fashion requiring copyright permission, other than the making of an
|
|
||||||
exact copy. The resulting work is called a "modified version" of the
|
|
||||||
earlier work or a work "based on" the earlier work.
|
|
||||||
|
|
||||||
A "covered work" means either the unmodified Program or a work based
|
|
||||||
on the Program.
|
|
||||||
|
|
||||||
To "propagate" a work means to do anything with it that, without
|
|
||||||
permission, would make you directly or secondarily liable for
|
|
||||||
infringement under applicable copyright law, except executing it on a
|
|
||||||
computer or modifying a private copy. Propagation includes copying,
|
|
||||||
distribution (with or without modification), making available to the
|
|
||||||
public, and in some countries other activities as well.
|
|
||||||
|
|
||||||
To "convey" a work means any kind of propagation that enables other
|
|
||||||
parties to make or receive copies. Mere interaction with a user through
|
|
||||||
a computer network, with no transfer of a copy, is not conveying.
|
|
||||||
|
|
||||||
An interactive user interface displays "Appropriate Legal Notices"
|
|
||||||
to the extent that it includes a convenient and prominently visible
|
|
||||||
feature that (1) displays an appropriate copyright notice, and (2)
|
|
||||||
tells the user that there is no warranty for the work (except to the
|
|
||||||
extent that warranties are provided), that licensees may convey the
|
|
||||||
work under this License, and how to view a copy of this License. If
|
|
||||||
the interface presents a list of user commands or options, such as a
|
|
||||||
menu, a prominent item in the list meets this criterion.
|
|
||||||
|
|
||||||
1. Source Code.
|
|
||||||
|
|
||||||
The "source code" for a work means the preferred form of the work
|
|
||||||
for making modifications to it. "Object code" means any non-source
|
|
||||||
form of a work.
|
|
||||||
|
|
||||||
A "Standard Interface" means an interface that either is an official
|
|
||||||
standard defined by a recognized standards body, or, in the case of
|
|
||||||
interfaces specified for a particular programming language, one that
|
|
||||||
is widely used among developers working in that language.
|
|
||||||
|
|
||||||
The "System Libraries" of an executable work include anything, other
|
|
||||||
than the work as a whole, that (a) is included in the normal form of
|
|
||||||
packaging a Major Component, but which is not part of that Major
|
|
||||||
Component, and (b) serves only to enable use of the work with that
|
|
||||||
Major Component, or to implement a Standard Interface for which an
|
|
||||||
implementation is available to the public in source code form. A
|
|
||||||
"Major Component", in this context, means a major essential component
|
|
||||||
(kernel, window system, and so on) of the specific operating system
|
|
||||||
(if any) on which the executable work runs, or a compiler used to
|
|
||||||
produce the work, or an object code interpreter used to run it.
|
|
||||||
|
|
||||||
The "Corresponding Source" for a work in object code form means all
|
|
||||||
the source code needed to generate, install, and (for an executable
|
|
||||||
work) run the object code and to modify the work, including scripts to
|
|
||||||
control those activities. However, it does not include the work's
|
|
||||||
System Libraries, or general-purpose tools or generally available free
|
|
||||||
programs which are used unmodified in performing those activities but
|
|
||||||
which are not part of the work. For example, Corresponding Source
|
|
||||||
includes interface definition files associated with source files for
|
|
||||||
the work, and the source code for shared libraries and dynamically
|
|
||||||
linked subprograms that the work is specifically designed to require,
|
|
||||||
such as by intimate data communication or control flow between those
|
|
||||||
subprograms and other parts of the work.
|
|
||||||
|
|
||||||
The Corresponding Source need not include anything that users
|
|
||||||
can regenerate automatically from other parts of the Corresponding
|
|
||||||
Source.
|
|
||||||
|
|
||||||
The Corresponding Source for a work in source code form is that
|
|
||||||
same work.
|
|
||||||
|
|
||||||
2. Basic Permissions.
|
|
||||||
|
|
||||||
All rights granted under this License are granted for the term of
|
|
||||||
copyright on the Program, and are irrevocable provided the stated
|
|
||||||
conditions are met. This License explicitly affirms your unlimited
|
|
||||||
permission to run the unmodified Program. The output from running a
|
|
||||||
covered work is covered by this License only if the output, given its
|
|
||||||
content, constitutes a covered work. This License acknowledges your
|
|
||||||
rights of fair use or other equivalent, as provided by copyright law.
|
|
||||||
|
|
||||||
You may make, run and propagate covered works that you do not
|
|
||||||
convey, without conditions so long as your license otherwise remains
|
|
||||||
in force. You may convey covered works to others for the sole purpose
|
|
||||||
of having them make modifications exclusively for you, or provide you
|
|
||||||
with facilities for running those works, provided that you comply with
|
|
||||||
the terms of this License in conveying all material for which you do
|
|
||||||
not control copyright. Those thus making or running the covered works
|
|
||||||
for you must do so exclusively on your behalf, under your direction
|
|
||||||
and control, on terms that prohibit them from making any copies of
|
|
||||||
your copyrighted material outside their relationship with you.
|
|
||||||
|
|
||||||
Conveying under any other circumstances is permitted solely under
|
|
||||||
the conditions stated below. Sublicensing is not allowed; section 10
|
|
||||||
makes it unnecessary.
|
|
||||||
|
|
||||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
|
||||||
|
|
||||||
No covered work shall be deemed part of an effective technological
|
|
||||||
measure under any applicable law fulfilling obligations under article
|
|
||||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
|
||||||
similar laws prohibiting or restricting circumvention of such
|
|
||||||
measures.
|
|
||||||
|
|
||||||
When you convey a covered work, you waive any legal power to forbid
|
|
||||||
circumvention of technological measures to the extent such circumvention
|
|
||||||
is effected by exercising rights under this License with respect to
|
|
||||||
the covered work, and you disclaim any intention to limit operation or
|
|
||||||
modification of the work as a means of enforcing, against the work's
|
|
||||||
users, your or third parties' legal rights to forbid circumvention of
|
|
||||||
technological measures.
|
|
||||||
|
|
||||||
4. Conveying Verbatim Copies.
|
|
||||||
|
|
||||||
You may convey verbatim copies of the Program's source code as you
|
|
||||||
receive it, in any medium, provided that you conspicuously and
|
|
||||||
appropriately publish on each copy an appropriate copyright notice;
|
|
||||||
keep intact all notices stating that this License and any
|
|
||||||
non-permissive terms added in accord with section 7 apply to the code;
|
|
||||||
keep intact all notices of the absence of any warranty; and give all
|
|
||||||
recipients a copy of this License along with the Program.
|
|
||||||
|
|
||||||
You may charge any price or no price for each copy that you convey,
|
|
||||||
and you may offer support or warranty protection for a fee.
|
|
||||||
|
|
||||||
5. Conveying Modified Source Versions.
|
|
||||||
|
|
||||||
You may convey a work based on the Program, or the modifications to
|
|
||||||
produce it from the Program, in the form of source code under the
|
|
||||||
terms of section 4, provided that you also meet all of these conditions:
|
|
||||||
|
|
||||||
a) The work must carry prominent notices stating that you modified
|
|
||||||
it, and giving a relevant date.
|
|
||||||
|
|
||||||
b) The work must carry prominent notices stating that it is
|
|
||||||
released under this License and any conditions added under section
|
|
||||||
7. This requirement modifies the requirement in section 4 to
|
|
||||||
"keep intact all notices".
|
|
||||||
|
|
||||||
c) You must license the entire work, as a whole, under this
|
|
||||||
License to anyone who comes into possession of a copy. This
|
|
||||||
License will therefore apply, along with any applicable section 7
|
|
||||||
additional terms, to the whole of the work, and all its parts,
|
|
||||||
regardless of how they are packaged. This License gives no
|
|
||||||
permission to license the work in any other way, but it does not
|
|
||||||
invalidate such permission if you have separately received it.
|
|
||||||
|
|
||||||
d) If the work has interactive user interfaces, each must display
|
|
||||||
Appropriate Legal Notices; however, if the Program has interactive
|
|
||||||
interfaces that do not display Appropriate Legal Notices, your
|
|
||||||
work need not make them do so.
|
|
||||||
|
|
||||||
A compilation of a covered work with other separate and independent
|
|
||||||
works, which are not by their nature extensions of the covered work,
|
|
||||||
and which are not combined with it such as to form a larger program,
|
|
||||||
in or on a volume of a storage or distribution medium, is called an
|
|
||||||
"aggregate" if the compilation and its resulting copyright are not
|
|
||||||
used to limit the access or legal rights of the compilation's users
|
|
||||||
beyond what the individual works permit. Inclusion of a covered work
|
|
||||||
in an aggregate does not cause this License to apply to the other
|
|
||||||
parts of the aggregate.
|
|
||||||
|
|
||||||
6. Conveying Non-Source Forms.
|
|
||||||
|
|
||||||
You may convey a covered work in object code form under the terms
|
|
||||||
of sections 4 and 5, provided that you also convey the
|
|
||||||
machine-readable Corresponding Source under the terms of this License,
|
|
||||||
in one of these ways:
|
|
||||||
|
|
||||||
a) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by the
|
|
||||||
Corresponding Source fixed on a durable physical medium
|
|
||||||
customarily used for software interchange.
|
|
||||||
|
|
||||||
b) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by a
|
|
||||||
written offer, valid for at least three years and valid for as
|
|
||||||
long as you offer spare parts or customer support for that product
|
|
||||||
model, to give anyone who possesses the object code either (1) a
|
|
||||||
copy of the Corresponding Source for all the software in the
|
|
||||||
product that is covered by this License, on a durable physical
|
|
||||||
medium customarily used for software interchange, for a price no
|
|
||||||
more than your reasonable cost of physically performing this
|
|
||||||
conveying of source, or (2) access to copy the
|
|
||||||
Corresponding Source from a network server at no charge.
|
|
||||||
|
|
||||||
c) Convey individual copies of the object code with a copy of the
|
|
||||||
written offer to provide the Corresponding Source. This
|
|
||||||
alternative is allowed only occasionally and noncommercially, and
|
|
||||||
only if you received the object code with such an offer, in accord
|
|
||||||
with subsection 6b.
|
|
||||||
|
|
||||||
d) Convey the object code by offering access from a designated
|
|
||||||
place (gratis or for a charge), and offer equivalent access to the
|
|
||||||
Corresponding Source in the same way through the same place at no
|
|
||||||
further charge. You need not require recipients to copy the
|
|
||||||
Corresponding Source along with the object code. If the place to
|
|
||||||
copy the object code is a network server, the Corresponding Source
|
|
||||||
may be on a different server (operated by you or a third party)
|
|
||||||
that supports equivalent copying facilities, provided you maintain
|
|
||||||
clear directions next to the object code saying where to find the
|
|
||||||
Corresponding Source. Regardless of what server hosts the
|
|
||||||
Corresponding Source, you remain obligated to ensure that it is
|
|
||||||
available for as long as needed to satisfy these requirements.
|
|
||||||
|
|
||||||
e) Convey the object code using peer-to-peer transmission, provided
|
|
||||||
you inform other peers where the object code and Corresponding
|
|
||||||
Source of the work are being offered to the general public at no
|
|
||||||
charge under subsection 6d.
|
|
||||||
|
|
||||||
A separable portion of the object code, whose source code is excluded
|
|
||||||
from the Corresponding Source as a System Library, need not be
|
|
||||||
included in conveying the object code work.
|
|
||||||
|
|
||||||
A "User Product" is either (1) a "consumer product", which means any
|
|
||||||
tangible personal property which is normally used for personal, family,
|
|
||||||
or household purposes, or (2) anything designed or sold for incorporation
|
|
||||||
into a dwelling. In determining whether a product is a consumer product,
|
|
||||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
|
||||||
product received by a particular user, "normally used" refers to a
|
|
||||||
typical or common use of that class of product, regardless of the status
|
|
||||||
of the particular user or of the way in which the particular user
|
|
||||||
actually uses, or expects or is expected to use, the product. A product
|
|
||||||
is a consumer product regardless of whether the product has substantial
|
|
||||||
commercial, industrial or non-consumer uses, unless such uses represent
|
|
||||||
the only significant mode of use of the product.
|
|
||||||
|
|
||||||
"Installation Information" for a User Product means any methods,
|
|
||||||
procedures, authorization keys, or other information required to install
|
|
||||||
and execute modified versions of a covered work in that User Product from
|
|
||||||
a modified version of its Corresponding Source. The information must
|
|
||||||
suffice to ensure that the continued functioning of the modified object
|
|
||||||
code is in no case prevented or interfered with solely because
|
|
||||||
modification has been made.
|
|
||||||
|
|
||||||
If you convey an object code work under this section in, or with, or
|
|
||||||
specifically for use in, a User Product, and the conveying occurs as
|
|
||||||
part of a transaction in which the right of possession and use of the
|
|
||||||
User Product is transferred to the recipient in perpetuity or for a
|
|
||||||
fixed term (regardless of how the transaction is characterized), the
|
|
||||||
Corresponding Source conveyed under this section must be accompanied
|
|
||||||
by the Installation Information. But this requirement does not apply
|
|
||||||
if neither you nor any third party retains the ability to install
|
|
||||||
modified object code on the User Product (for example, the work has
|
|
||||||
been installed in ROM).
|
|
||||||
|
|
||||||
The requirement to provide Installation Information does not include a
|
|
||||||
requirement to continue to provide support service, warranty, or updates
|
|
||||||
for a work that has been modified or installed by the recipient, or for
|
|
||||||
the User Product in which it has been modified or installed. Access to a
|
|
||||||
network may be denied when the modification itself materially and
|
|
||||||
adversely affects the operation of the network or violates the rules and
|
|
||||||
protocols for communication across the network.
|
|
||||||
|
|
||||||
Corresponding Source conveyed, and Installation Information provided,
|
|
||||||
in accord with this section must be in a format that is publicly
|
|
||||||
documented (and with an implementation available to the public in
|
|
||||||
source code form), and must require no special password or key for
|
|
||||||
unpacking, reading or copying.
|
|
||||||
|
|
||||||
7. Additional Terms.
|
|
||||||
|
|
||||||
"Additional permissions" are terms that supplement the terms of this
|
|
||||||
License by making exceptions from one or more of its conditions.
|
|
||||||
Additional permissions that are applicable to the entire Program shall
|
|
||||||
be treated as though they were included in this License, to the extent
|
|
||||||
that they are valid under applicable law. If additional permissions
|
|
||||||
apply only to part of the Program, that part may be used separately
|
|
||||||
under those permissions, but the entire Program remains governed by
|
|
||||||
this License without regard to the additional permissions.
|
|
||||||
|
|
||||||
When you convey a copy of a covered work, you may at your option
|
|
||||||
remove any additional permissions from that copy, or from any part of
|
|
||||||
it. (Additional permissions may be written to require their own
|
|
||||||
removal in certain cases when you modify the work.) You may place
|
|
||||||
additional permissions on material, added by you to a covered work,
|
|
||||||
for which you have or can give appropriate copyright permission.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, for material you
|
|
||||||
add to a covered work, you may (if authorized by the copyright holders of
|
|
||||||
that material) supplement the terms of this License with terms:
|
|
||||||
|
|
||||||
a) Disclaiming warranty or limiting liability differently from the
|
|
||||||
terms of sections 15 and 16 of this License; or
|
|
||||||
|
|
||||||
b) Requiring preservation of specified reasonable legal notices or
|
|
||||||
author attributions in that material or in the Appropriate Legal
|
|
||||||
Notices displayed by works containing it; or
|
|
||||||
|
|
||||||
c) Prohibiting misrepresentation of the origin of that material, or
|
|
||||||
requiring that modified versions of such material be marked in
|
|
||||||
reasonable ways as different from the original version; or
|
|
||||||
|
|
||||||
d) Limiting the use for publicity purposes of names of licensors or
|
|
||||||
authors of the material; or
|
|
||||||
|
|
||||||
e) Declining to grant rights under trademark law for use of some
|
|
||||||
trade names, trademarks, or service marks; or
|
|
||||||
|
|
||||||
f) Requiring indemnification of licensors and authors of that
|
|
||||||
material by anyone who conveys the material (or modified versions of
|
|
||||||
it) with contractual assumptions of liability to the recipient, for
|
|
||||||
any liability that these contractual assumptions directly impose on
|
|
||||||
those licensors and authors.
|
|
||||||
|
|
||||||
All other non-permissive additional terms are considered "further
|
|
||||||
restrictions" within the meaning of section 10. If the Program as you
|
|
||||||
received it, or any part of it, contains a notice stating that it is
|
|
||||||
governed by this License along with a term that is a further
|
|
||||||
restriction, you may remove that term. If a license document contains
|
|
||||||
a further restriction but permits relicensing or conveying under this
|
|
||||||
License, you may add to a covered work material governed by the terms
|
|
||||||
of that license document, provided that the further restriction does
|
|
||||||
not survive such relicensing or conveying.
|
|
||||||
|
|
||||||
If you add terms to a covered work in accord with this section, you
|
|
||||||
must place, in the relevant source files, a statement of the
|
|
||||||
additional terms that apply to those files, or a notice indicating
|
|
||||||
where to find the applicable terms.
|
|
||||||
|
|
||||||
Additional terms, permissive or non-permissive, may be stated in the
|
|
||||||
form of a separately written license, or stated as exceptions;
|
|
||||||
the above requirements apply either way.
|
|
||||||
|
|
||||||
8. Termination.
|
|
||||||
|
|
||||||
You may not propagate or modify a covered work except as expressly
|
|
||||||
provided under this License. Any attempt otherwise to propagate or
|
|
||||||
modify it is void, and will automatically terminate your rights under
|
|
||||||
this License (including any patent licenses granted under the third
|
|
||||||
paragraph of section 11).
|
|
||||||
|
|
||||||
However, if you cease all violation of this License, then your
|
|
||||||
license from a particular copyright holder is reinstated (a)
|
|
||||||
provisionally, unless and until the copyright holder explicitly and
|
|
||||||
finally terminates your license, and (b) permanently, if the copyright
|
|
||||||
holder fails to notify you of the violation by some reasonable means
|
|
||||||
prior to 60 days after the cessation.
|
|
||||||
|
|
||||||
Moreover, your license from a particular copyright holder is
|
|
||||||
reinstated permanently if the copyright holder notifies you of the
|
|
||||||
violation by some reasonable means, this is the first time you have
|
|
||||||
received notice of violation of this License (for any work) from that
|
|
||||||
copyright holder, and you cure the violation prior to 30 days after
|
|
||||||
your receipt of the notice.
|
|
||||||
|
|
||||||
Termination of your rights under this section does not terminate the
|
|
||||||
licenses of parties who have received copies or rights from you under
|
|
||||||
this License. If your rights have been terminated and not permanently
|
|
||||||
reinstated, you do not qualify to receive new licenses for the same
|
|
||||||
material under section 10.
|
|
||||||
|
|
||||||
9. Acceptance Not Required for Having Copies.
|
|
||||||
|
|
||||||
You are not required to accept this License in order to receive or
|
|
||||||
run a copy of the Program. Ancillary propagation of a covered work
|
|
||||||
occurring solely as a consequence of using peer-to-peer transmission
|
|
||||||
to receive a copy likewise does not require acceptance. However,
|
|
||||||
nothing other than this License grants you permission to propagate or
|
|
||||||
modify any covered work. These actions infringe copyright if you do
|
|
||||||
not accept this License. Therefore, by modifying or propagating a
|
|
||||||
covered work, you indicate your acceptance of this License to do so.
|
|
||||||
|
|
||||||
10. Automatic Licensing of Downstream Recipients.
|
|
||||||
|
|
||||||
Each time you convey a covered work, the recipient automatically
|
|
||||||
receives a license from the original licensors, to run, modify and
|
|
||||||
propagate that work, subject to this License. You are not responsible
|
|
||||||
for enforcing compliance by third parties with this License.
|
|
||||||
|
|
||||||
An "entity transaction" is a transaction transferring control of an
|
|
||||||
organization, or substantially all assets of one, or subdividing an
|
|
||||||
organization, or merging organizations. If propagation of a covered
|
|
||||||
work results from an entity transaction, each party to that
|
|
||||||
transaction who receives a copy of the work also receives whatever
|
|
||||||
licenses to the work the party's predecessor in interest had or could
|
|
||||||
give under the previous paragraph, plus a right to possession of the
|
|
||||||
Corresponding Source of the work from the predecessor in interest, if
|
|
||||||
the predecessor has it or can get it with reasonable efforts.
|
|
||||||
|
|
||||||
You may not impose any further restrictions on the exercise of the
|
|
||||||
rights granted or affirmed under this License. For example, you may
|
|
||||||
not impose a license fee, royalty, or other charge for exercise of
|
|
||||||
rights granted under this License, and you may not initiate litigation
|
|
||||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
|
||||||
any patent claim is infringed by making, using, selling, offering for
|
|
||||||
sale, or importing the Program or any portion of it.
|
|
||||||
|
|
||||||
11. Patents.
|
|
||||||
|
|
||||||
A "contributor" is a copyright holder who authorizes use under this
|
|
||||||
License of the Program or a work on which the Program is based. The
|
|
||||||
work thus licensed is called the contributor's "contributor version".
|
|
||||||
|
|
||||||
A contributor's "essential patent claims" are all patent claims
|
|
||||||
owned or controlled by the contributor, whether already acquired or
|
|
||||||
hereafter acquired, that would be infringed by some manner, permitted
|
|
||||||
by this License, of making, using, or selling its contributor version,
|
|
||||||
but do not include claims that would be infringed only as a
|
|
||||||
consequence of further modification of the contributor version. For
|
|
||||||
purposes of this definition, "control" includes the right to grant
|
|
||||||
patent sublicenses in a manner consistent with the requirements of
|
|
||||||
this License.
|
|
||||||
|
|
||||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
|
||||||
patent license under the contributor's essential patent claims, to
|
|
||||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
|
||||||
propagate the contents of its contributor version.
|
|
||||||
|
|
||||||
In the following three paragraphs, a "patent license" is any express
|
|
||||||
agreement or commitment, however denominated, not to enforce a patent
|
|
||||||
(such as an express permission to practice a patent or covenant not to
|
|
||||||
sue for patent infringement). To "grant" such a patent license to a
|
|
||||||
party means to make such an agreement or commitment not to enforce a
|
|
||||||
patent against the party.
|
|
||||||
|
|
||||||
If you convey a covered work, knowingly relying on a patent license,
|
|
||||||
and the Corresponding Source of the work is not available for anyone
|
|
||||||
to copy, free of charge and under the terms of this License, through a
|
|
||||||
publicly available network server or other readily accessible means,
|
|
||||||
then you must either (1) cause the Corresponding Source to be so
|
|
||||||
available, or (2) arrange to deprive yourself of the benefit of the
|
|
||||||
patent license for this particular work, or (3) arrange, in a manner
|
|
||||||
consistent with the requirements of this License, to extend the patent
|
|
||||||
license to downstream recipients. "Knowingly relying" means you have
|
|
||||||
actual knowledge that, but for the patent license, your conveying the
|
|
||||||
covered work in a country, or your recipient's use of the covered work
|
|
||||||
in a country, would infringe one or more identifiable patents in that
|
|
||||||
country that you have reason to believe are valid.
|
|
||||||
|
|
||||||
If, pursuant to or in connection with a single transaction or
|
|
||||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
|
||||||
covered work, and grant a patent license to some of the parties
|
|
||||||
receiving the covered work authorizing them to use, propagate, modify
|
|
||||||
or convey a specific copy of the covered work, then the patent license
|
|
||||||
you grant is automatically extended to all recipients of the covered
|
|
||||||
work and works based on it.
|
|
||||||
|
|
||||||
A patent license is "discriminatory" if it does not include within
|
|
||||||
the scope of its coverage, prohibits the exercise of, or is
|
|
||||||
conditioned on the non-exercise of one or more of the rights that are
|
|
||||||
specifically granted under this License. You may not convey a covered
|
|
||||||
work if you are a party to an arrangement with a third party that is
|
|
||||||
in the business of distributing software, under which you make payment
|
|
||||||
to the third party based on the extent of your activity of conveying
|
|
||||||
the work, and under which the third party grants, to any of the
|
|
||||||
parties who would receive the covered work from you, a discriminatory
|
|
||||||
patent license (a) in connection with copies of the covered work
|
|
||||||
conveyed by you (or copies made from those copies), or (b) primarily
|
|
||||||
for and in connection with specific products or compilations that
|
|
||||||
contain the covered work, unless you entered into that arrangement,
|
|
||||||
or that patent license was granted, prior to 28 March 2007.
|
|
||||||
|
|
||||||
Nothing in this License shall be construed as excluding or limiting
|
|
||||||
any implied license or other defenses to infringement that may
|
|
||||||
otherwise be available to you under applicable patent law.
|
|
||||||
|
|
||||||
12. No Surrender of Others' Freedom.
|
|
||||||
|
|
||||||
If conditions are imposed on you (whether by court order, agreement or
|
|
||||||
otherwise) that contradict the conditions of this License, they do not
|
|
||||||
excuse you from the conditions of this License. If you cannot convey a
|
|
||||||
covered work so as to satisfy simultaneously your obligations under this
|
|
||||||
License and any other pertinent obligations, then as a consequence you may
|
|
||||||
not convey it at all. For example, if you agree to terms that obligate you
|
|
||||||
to collect a royalty for further conveying from those to whom you convey
|
|
||||||
the Program, the only way you could satisfy both those terms and this
|
|
||||||
License would be to refrain entirely from conveying the Program.
|
|
||||||
|
|
||||||
13. Remote Network Interaction; Use with the GNU General Public License.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, if you modify the
|
|
||||||
Program, your modified version must prominently offer all users
|
|
||||||
interacting with it remotely through a computer network (if your version
|
|
||||||
supports such interaction) an opportunity to receive the Corresponding
|
|
||||||
Source of your version by providing access to the Corresponding Source
|
|
||||||
from a network server at no charge, through some standard or customary
|
|
||||||
means of facilitating copying of software. This Corresponding Source
|
|
||||||
shall include the Corresponding Source for any work covered by version 3
|
|
||||||
of the GNU General Public License that is incorporated pursuant to the
|
|
||||||
following paragraph.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, you have
|
|
||||||
permission to link or combine any covered work with a work licensed
|
|
||||||
under version 3 of the GNU General Public License into a single
|
|
||||||
combined work, and to convey the resulting work. The terms of this
|
|
||||||
License will continue to apply to the part which is the covered work,
|
|
||||||
but the work with which it is combined will remain governed by version
|
|
||||||
3 of the GNU General Public License.
|
|
||||||
|
|
||||||
14. Revised Versions of this License.
|
|
||||||
|
|
||||||
The Free Software Foundation may publish revised and/or new versions of
|
|
||||||
the GNU Affero General Public License from time to time. Such new versions
|
|
||||||
will be similar in spirit to the present version, but may differ in detail to
|
|
||||||
address new problems or concerns.
|
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the
|
|
||||||
Program specifies that a certain numbered version of the GNU Affero General
|
|
||||||
Public License "or any later version" applies to it, you have the
|
|
||||||
option of following the terms and conditions either of that numbered
|
|
||||||
version or of any later version published by the Free Software
|
|
||||||
Foundation. If the Program does not specify a version number of the
|
|
||||||
GNU Affero General Public License, you may choose any version ever published
|
|
||||||
by the Free Software Foundation.
|
|
||||||
|
|
||||||
If the Program specifies that a proxy can decide which future
|
|
||||||
versions of the GNU Affero General Public License can be used, that proxy's
|
|
||||||
public statement of acceptance of a version permanently authorizes you
|
|
||||||
to choose that version for the Program.
|
|
||||||
|
|
||||||
Later license versions may give you additional or different
|
|
||||||
permissions. However, no additional obligations are imposed on any
|
|
||||||
author or copyright holder as a result of your choosing to follow a
|
|
||||||
later version.
|
|
||||||
|
|
||||||
15. Disclaimer of Warranty.
|
|
||||||
|
|
||||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
|
||||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
|
||||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
|
||||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
|
||||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
||||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
|
||||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
|
||||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
|
||||||
|
|
||||||
16. Limitation of Liability.
|
|
||||||
|
|
||||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
|
||||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
|
||||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
|
||||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
|
||||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
|
||||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
|
||||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
|
||||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
|
||||||
SUCH DAMAGES.
|
|
||||||
|
|
||||||
17. Interpretation of Sections 15 and 16.
|
|
||||||
|
|
||||||
If the disclaimer of warranty and limitation of liability provided
|
|
||||||
above cannot be given local legal effect according to their terms,
|
|
||||||
reviewing courts shall apply local law that most closely approximates
|
|
||||||
an absolute waiver of all civil liability in connection with the
|
|
||||||
Program, unless a warranty or assumption of liability accompanies a
|
|
||||||
copy of the Program in return for a fee.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
How to Apply These Terms to Your New Programs
|
|
||||||
|
|
||||||
If you develop a new program, and you want it to be of the greatest
|
|
||||||
possible use to the public, the best way to achieve this is to make it
|
|
||||||
free software which everyone can redistribute and change under these terms.
|
|
||||||
|
|
||||||
To do so, attach the following notices to the program. It is safest
|
|
||||||
to attach them to the start of each source file to most effectively
|
|
||||||
state the exclusion of warranty; and each file should have at least
|
|
||||||
the "copyright" line and a pointer to where the full notice is found.
|
|
||||||
|
|
||||||
<one line to give the program's name and a brief idea of what it does.>
|
|
||||||
Copyright (C) <year> <name of author>
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU Affero General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU Affero General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper mail.
|
|
||||||
|
|
||||||
If your software can interact with users remotely through a computer
|
|
||||||
network, you should also make sure that it provides a way for users to
|
|
||||||
get its source. For example, if your program is a web application, its
|
|
||||||
interface could display a "Source" link that leads users to an archive
|
|
||||||
of the code. There are many ways you could offer source, and different
|
|
||||||
solutions will be better for different programs; see section 13 for the
|
|
||||||
specific requirements.
|
|
||||||
|
|
||||||
You should also get your employer (if you work as a programmer) or school,
|
|
||||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
|
||||||
For more information on this, and how to apply and follow the GNU AGPL, see
|
|
||||||
<http://www.gnu.org/licenses/>.
|
|
16
README.md
@ -1,4 +1,14 @@
|
|||||||
humhub
|
Welcome to HumHub
|
||||||
======
|
=================
|
||||||
|
|
||||||
|
HumHub is an opensource social network application written in PHP.
|
||||||
|
|
||||||
|
- <a href="http://demo.humhub.org" target="_blank">**Demo**</a>
|
||||||
|
|
||||||
|
- <a href="protected/docs/guide/administration/index.md">Installation & Admin Documentation</a>
|
||||||
|
- <a href="protected/docs/guide/developer/index.md">Developer Documentation</a>
|
||||||
|
- <a href="protected/docs/guide/theming/index.md">Theming Documentation</a>
|
||||||
|
|
||||||
|
|
||||||
|
- <a href="protected/docs/license.md">License: Dual license AGPL v3 / Proprietary</a>
|
||||||
|
|
||||||
HumHub - OpenSource Social Network
|
|
||||||
|
0
assets/empty.txt
Normal file
102
css/bootstrap-wysihtml5.css
vendored
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
ul.wysihtml5-toolbar {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar::after {
|
||||||
|
clear: both;
|
||||||
|
display: table;
|
||||||
|
content: "";
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar > li {
|
||||||
|
float: left;
|
||||||
|
display: list-item;
|
||||||
|
list-style: none;
|
||||||
|
margin: 0 5px 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar a[data-wysihtml5-command=bold] {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar a[data-wysihtml5-command=italic] {
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar a[data-wysihtml5-command=underline] {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar a.btn.wysihtml5-command-active {
|
||||||
|
background-image: none;
|
||||||
|
-webkit-box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15),0 1px 2px rgba(0, 0, 0, 0.05);
|
||||||
|
-moz-box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15),0 1px 2px rgba(0, 0, 0, 0.05);
|
||||||
|
box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15),0 1px 2px rgba(0, 0, 0, 0.05);
|
||||||
|
background-color: #E6E6E6;
|
||||||
|
background-color: #D9D9D9;
|
||||||
|
outline: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-commands-disabled .dropdown-menu {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div.wysihtml5-colors {
|
||||||
|
display:block;
|
||||||
|
width: 50px;
|
||||||
|
height: 20px;
|
||||||
|
margin-top: 2px;
|
||||||
|
margin-left: 5px;
|
||||||
|
position: absolute;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar a.wysihtml5-colors-title {
|
||||||
|
padding-left: 70px;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="black"] {
|
||||||
|
background: black !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="silver"] {
|
||||||
|
background: silver !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="gray"] {
|
||||||
|
background: gray !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="maroon"] {
|
||||||
|
background: maroon !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="red"] {
|
||||||
|
background: red !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="purple"] {
|
||||||
|
background: purple !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="green"] {
|
||||||
|
background: green !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="olive"] {
|
||||||
|
background: olive !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="navy"] {
|
||||||
|
background: navy !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="blue"] {
|
||||||
|
background: blue !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.wysihtml5-toolbar div[data-wysihtml5-command-value="orange"] {
|
||||||
|
background: orange !important;
|
||||||
|
}
|
9
css/bootstrap.min.css
vendored
Normal file
67
css/bootstrap3-wysiwyg5-color.css
vendored
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
.wysiwyg-color-black {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-silver {
|
||||||
|
color: silver;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-gray {
|
||||||
|
color: gray;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-white {
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-maroon {
|
||||||
|
color: maroon;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-red {
|
||||||
|
color: red;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-purple {
|
||||||
|
color: purple;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-fuchsia {
|
||||||
|
color: fuchsia;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-green {
|
||||||
|
color: green;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-lime {
|
||||||
|
color: lime;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-olive {
|
||||||
|
color: olive;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-yellow {
|
||||||
|
color: yellow;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-navy {
|
||||||
|
color: navy;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-blue {
|
||||||
|
color: blue;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-teal {
|
||||||
|
color: teal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-aqua {
|
||||||
|
color: aqua;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wysiwyg-color-orange {
|
||||||
|
color: orange;
|
||||||
|
}
|
481
css/datepicker.css
Normal file
@ -0,0 +1,481 @@
|
|||||||
|
/*!
|
||||||
|
* Datepicker for Bootstrap
|
||||||
|
*
|
||||||
|
* Copyright 2012 Stefan Petre
|
||||||
|
* Improvements by Andrew Rowls
|
||||||
|
* Licensed under the Apache License v2.0
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
.datepicker {
|
||||||
|
padding: 4px;
|
||||||
|
-webkit-border-radius: 4px;
|
||||||
|
-moz-border-radius: 4px;
|
||||||
|
border-radius: 4px;
|
||||||
|
direction: ltr;
|
||||||
|
/*.dow {
|
||||||
|
border-top: 1px solid #ddd !important;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
}
|
||||||
|
.datepicker-inline {
|
||||||
|
width: 220px;
|
||||||
|
}
|
||||||
|
.datepicker.datepicker-rtl {
|
||||||
|
direction: rtl;
|
||||||
|
}
|
||||||
|
.datepicker.datepicker-rtl table tr td span {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown {
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown:before {
|
||||||
|
content: '';
|
||||||
|
display: inline-block;
|
||||||
|
border-left: 7px solid transparent;
|
||||||
|
border-right: 7px solid transparent;
|
||||||
|
border-bottom: 7px solid #ccc;
|
||||||
|
border-top: 0;
|
||||||
|
border-bottom-color: rgba(0, 0, 0, 0.2);
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown:after {
|
||||||
|
content: '';
|
||||||
|
display: inline-block;
|
||||||
|
border-left: 6px solid transparent;
|
||||||
|
border-right: 6px solid transparent;
|
||||||
|
border-bottom: 6px solid #ffffff;
|
||||||
|
border-top: 0;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-left:before {
|
||||||
|
left: 6px;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-left:after {
|
||||||
|
left: 7px;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-right:before {
|
||||||
|
right: 6px;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-right:after {
|
||||||
|
right: 7px;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-top:before {
|
||||||
|
top: -7px;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-top:after {
|
||||||
|
top: -6px;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-bottom:before {
|
||||||
|
bottom: -7px;
|
||||||
|
border-bottom: 0;
|
||||||
|
border-top: 7px solid #999;
|
||||||
|
}
|
||||||
|
.datepicker-dropdown.datepicker-orient-bottom:after {
|
||||||
|
bottom: -6px;
|
||||||
|
border-bottom: 0;
|
||||||
|
border-top: 6px solid #ffffff;
|
||||||
|
}
|
||||||
|
.datepicker > div {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.datepicker.days div.datepicker-days {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.datepicker.months div.datepicker-months {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.datepicker.years div.datepicker-years {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.datepicker table {
|
||||||
|
margin: 0;
|
||||||
|
-webkit-touch-callout: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
-khtml-user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
-ms-user-select: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
.datepicker td,
|
||||||
|
.datepicker th {
|
||||||
|
text-align: center;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
-webkit-border-radius: 4px;
|
||||||
|
-moz-border-radius: 4px;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
.table-striped .datepicker table tr td,
|
||||||
|
.table-striped .datepicker table tr th {
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.day:hover {
|
||||||
|
background: #eeeeee;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.old,
|
||||||
|
.datepicker table tr td.new {
|
||||||
|
color: #999999;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.disabled,
|
||||||
|
.datepicker table tr td.disabled:hover {
|
||||||
|
background: none;
|
||||||
|
color: #999999;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.today,
|
||||||
|
.datepicker table tr td.today:hover,
|
||||||
|
.datepicker table tr td.today.disabled,
|
||||||
|
.datepicker table tr td.today.disabled:hover {
|
||||||
|
background-color: #fde19a;
|
||||||
|
background-image: -moz-linear-gradient(top, #fdd49a, #fdf59a);
|
||||||
|
background-image: -ms-linear-gradient(top, #fdd49a, #fdf59a);
|
||||||
|
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#fdd49a), to(#fdf59a));
|
||||||
|
background-image: -webkit-linear-gradient(top, #fdd49a, #fdf59a);
|
||||||
|
background-image: -o-linear-gradient(top, #fdd49a, #fdf59a);
|
||||||
|
background-image: linear-gradient(top, #fdd49a, #fdf59a);
|
||||||
|
background-repeat: repeat-x;
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fdd49a', endColorstr='#fdf59a', GradientType=0);
|
||||||
|
border-color: #fdf59a #fdf59a #fbed50;
|
||||||
|
border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.today:hover,
|
||||||
|
.datepicker table tr td.today:hover:hover,
|
||||||
|
.datepicker table tr td.today.disabled:hover,
|
||||||
|
.datepicker table tr td.today.disabled:hover:hover,
|
||||||
|
.datepicker table tr td.today:active,
|
||||||
|
.datepicker table tr td.today:hover:active,
|
||||||
|
.datepicker table tr td.today.disabled:active,
|
||||||
|
.datepicker table tr td.today.disabled:hover:active,
|
||||||
|
.datepicker table tr td.today.active,
|
||||||
|
.datepicker table tr td.today:hover.active,
|
||||||
|
.datepicker table tr td.today.disabled.active,
|
||||||
|
.datepicker table tr td.today.disabled:hover.active,
|
||||||
|
.datepicker table tr td.today.disabled,
|
||||||
|
.datepicker table tr td.today:hover.disabled,
|
||||||
|
.datepicker table tr td.today.disabled.disabled,
|
||||||
|
.datepicker table tr td.today.disabled:hover.disabled,
|
||||||
|
.datepicker table tr td.today[disabled],
|
||||||
|
.datepicker table tr td.today:hover[disabled],
|
||||||
|
.datepicker table tr td.today.disabled[disabled],
|
||||||
|
.datepicker table tr td.today.disabled:hover[disabled] {
|
||||||
|
background-color: #fdf59a;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.today:active,
|
||||||
|
.datepicker table tr td.today:hover:active,
|
||||||
|
.datepicker table tr td.today.disabled:active,
|
||||||
|
.datepicker table tr td.today.disabled:hover:active,
|
||||||
|
.datepicker table tr td.today.active,
|
||||||
|
.datepicker table tr td.today:hover.active,
|
||||||
|
.datepicker table tr td.today.disabled.active,
|
||||||
|
.datepicker table tr td.today.disabled:hover.active {
|
||||||
|
background-color: #fbf069 \9;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.today:hover:hover {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.today.active:hover {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.range,
|
||||||
|
.datepicker table tr td.range:hover,
|
||||||
|
.datepicker table tr td.range.disabled,
|
||||||
|
.datepicker table tr td.range.disabled:hover {
|
||||||
|
background: #eeeeee;
|
||||||
|
-webkit-border-radius: 0;
|
||||||
|
-moz-border-radius: 0;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.range.today,
|
||||||
|
.datepicker table tr td.range.today:hover,
|
||||||
|
.datepicker table tr td.range.today.disabled,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover {
|
||||||
|
background-color: #f3d17a;
|
||||||
|
background-image: -moz-linear-gradient(top, #f3c17a, #f3e97a);
|
||||||
|
background-image: -ms-linear-gradient(top, #f3c17a, #f3e97a);
|
||||||
|
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#f3c17a), to(#f3e97a));
|
||||||
|
background-image: -webkit-linear-gradient(top, #f3c17a, #f3e97a);
|
||||||
|
background-image: -o-linear-gradient(top, #f3c17a, #f3e97a);
|
||||||
|
background-image: linear-gradient(top, #f3c17a, #f3e97a);
|
||||||
|
background-repeat: repeat-x;
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#f3c17a', endColorstr='#f3e97a', GradientType=0);
|
||||||
|
border-color: #f3e97a #f3e97a #edde34;
|
||||||
|
border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
|
||||||
|
-webkit-border-radius: 0;
|
||||||
|
-moz-border-radius: 0;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.range.today:hover,
|
||||||
|
.datepicker table tr td.range.today:hover:hover,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover:hover,
|
||||||
|
.datepicker table tr td.range.today:active,
|
||||||
|
.datepicker table tr td.range.today:hover:active,
|
||||||
|
.datepicker table tr td.range.today.disabled:active,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover:active,
|
||||||
|
.datepicker table tr td.range.today.active,
|
||||||
|
.datepicker table tr td.range.today:hover.active,
|
||||||
|
.datepicker table tr td.range.today.disabled.active,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover.active,
|
||||||
|
.datepicker table tr td.range.today.disabled,
|
||||||
|
.datepicker table tr td.range.today:hover.disabled,
|
||||||
|
.datepicker table tr td.range.today.disabled.disabled,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover.disabled,
|
||||||
|
.datepicker table tr td.range.today[disabled],
|
||||||
|
.datepicker table tr td.range.today:hover[disabled],
|
||||||
|
.datepicker table tr td.range.today.disabled[disabled],
|
||||||
|
.datepicker table tr td.range.today.disabled:hover[disabled] {
|
||||||
|
background-color: #f3e97a;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.range.today:active,
|
||||||
|
.datepicker table tr td.range.today:hover:active,
|
||||||
|
.datepicker table tr td.range.today.disabled:active,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover:active,
|
||||||
|
.datepicker table tr td.range.today.active,
|
||||||
|
.datepicker table tr td.range.today:hover.active,
|
||||||
|
.datepicker table tr td.range.today.disabled.active,
|
||||||
|
.datepicker table tr td.range.today.disabled:hover.active {
|
||||||
|
background-color: #efe24b \9;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.selected,
|
||||||
|
.datepicker table tr td.selected:hover,
|
||||||
|
.datepicker table tr td.selected.disabled,
|
||||||
|
.datepicker table tr td.selected.disabled:hover {
|
||||||
|
background-color: #9e9e9e;
|
||||||
|
background-image: -moz-linear-gradient(top, #b3b3b3, #808080);
|
||||||
|
background-image: -ms-linear-gradient(top, #b3b3b3, #808080);
|
||||||
|
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#b3b3b3), to(#808080));
|
||||||
|
background-image: -webkit-linear-gradient(top, #b3b3b3, #808080);
|
||||||
|
background-image: -o-linear-gradient(top, #b3b3b3, #808080);
|
||||||
|
background-image: linear-gradient(top, #b3b3b3, #808080);
|
||||||
|
background-repeat: repeat-x;
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#b3b3b3', endColorstr='#808080', GradientType=0);
|
||||||
|
border-color: #808080 #808080 #595959;
|
||||||
|
border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
|
||||||
|
color: #fff;
|
||||||
|
text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.25);
|
||||||
|
}
|
||||||
|
.datepicker table tr td.selected:hover,
|
||||||
|
.datepicker table tr td.selected:hover:hover,
|
||||||
|
.datepicker table tr td.selected.disabled:hover,
|
||||||
|
.datepicker table tr td.selected.disabled:hover:hover,
|
||||||
|
.datepicker table tr td.selected:active,
|
||||||
|
.datepicker table tr td.selected:hover:active,
|
||||||
|
.datepicker table tr td.selected.disabled:active,
|
||||||
|
.datepicker table tr td.selected.disabled:hover:active,
|
||||||
|
.datepicker table tr td.selected.active,
|
||||||
|
.datepicker table tr td.selected:hover.active,
|
||||||
|
.datepicker table tr td.selected.disabled.active,
|
||||||
|
.datepicker table tr td.selected.disabled:hover.active,
|
||||||
|
.datepicker table tr td.selected.disabled,
|
||||||
|
.datepicker table tr td.selected:hover.disabled,
|
||||||
|
.datepicker table tr td.selected.disabled.disabled,
|
||||||
|
.datepicker table tr td.selected.disabled:hover.disabled,
|
||||||
|
.datepicker table tr td.selected[disabled],
|
||||||
|
.datepicker table tr td.selected:hover[disabled],
|
||||||
|
.datepicker table tr td.selected.disabled[disabled],
|
||||||
|
.datepicker table tr td.selected.disabled:hover[disabled] {
|
||||||
|
background-color: #808080;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.selected:active,
|
||||||
|
.datepicker table tr td.selected:hover:active,
|
||||||
|
.datepicker table tr td.selected.disabled:active,
|
||||||
|
.datepicker table tr td.selected.disabled:hover:active,
|
||||||
|
.datepicker table tr td.selected.active,
|
||||||
|
.datepicker table tr td.selected:hover.active,
|
||||||
|
.datepicker table tr td.selected.disabled.active,
|
||||||
|
.datepicker table tr td.selected.disabled:hover.active {
|
||||||
|
background-color: #666666 \9;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.active,
|
||||||
|
.datepicker table tr td.active:hover,
|
||||||
|
.datepicker table tr td.active.disabled,
|
||||||
|
.datepicker table tr td.active.disabled:hover {
|
||||||
|
background-color: #006dcc;
|
||||||
|
background-image: -moz-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: -ms-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#0088cc), to(#0044cc));
|
||||||
|
background-image: -webkit-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: -o-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-repeat: repeat-x;
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#0088cc', endColorstr='#0044cc', GradientType=0);
|
||||||
|
border-color: #0044cc #0044cc #002a80;
|
||||||
|
border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
|
||||||
|
color: #fff;
|
||||||
|
text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.25);
|
||||||
|
}
|
||||||
|
.datepicker table tr td.active:hover,
|
||||||
|
.datepicker table tr td.active:hover:hover,
|
||||||
|
.datepicker table tr td.active.disabled:hover,
|
||||||
|
.datepicker table tr td.active.disabled:hover:hover,
|
||||||
|
.datepicker table tr td.active:active,
|
||||||
|
.datepicker table tr td.active:hover:active,
|
||||||
|
.datepicker table tr td.active.disabled:active,
|
||||||
|
.datepicker table tr td.active.disabled:hover:active,
|
||||||
|
.datepicker table tr td.active.active,
|
||||||
|
.datepicker table tr td.active:hover.active,
|
||||||
|
.datepicker table tr td.active.disabled.active,
|
||||||
|
.datepicker table tr td.active.disabled:hover.active,
|
||||||
|
.datepicker table tr td.active.disabled,
|
||||||
|
.datepicker table tr td.active:hover.disabled,
|
||||||
|
.datepicker table tr td.active.disabled.disabled,
|
||||||
|
.datepicker table tr td.active.disabled:hover.disabled,
|
||||||
|
.datepicker table tr td.active[disabled],
|
||||||
|
.datepicker table tr td.active:hover[disabled],
|
||||||
|
.datepicker table tr td.active.disabled[disabled],
|
||||||
|
.datepicker table tr td.active.disabled:hover[disabled] {
|
||||||
|
background-color: #0044cc;
|
||||||
|
}
|
||||||
|
.datepicker table tr td.active:active,
|
||||||
|
.datepicker table tr td.active:hover:active,
|
||||||
|
.datepicker table tr td.active.disabled:active,
|
||||||
|
.datepicker table tr td.active.disabled:hover:active,
|
||||||
|
.datepicker table tr td.active.active,
|
||||||
|
.datepicker table tr td.active:hover.active,
|
||||||
|
.datepicker table tr td.active.disabled.active,
|
||||||
|
.datepicker table tr td.active.disabled:hover.active {
|
||||||
|
background-color: #003399 \9;
|
||||||
|
}
|
||||||
|
.datepicker table tr td span {
|
||||||
|
display: block;
|
||||||
|
width: 23%;
|
||||||
|
height: 54px;
|
||||||
|
line-height: 54px;
|
||||||
|
float: left;
|
||||||
|
margin: 1%;
|
||||||
|
cursor: pointer;
|
||||||
|
-webkit-border-radius: 4px;
|
||||||
|
-moz-border-radius: 4px;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
.datepicker table tr td span:hover {
|
||||||
|
background: #eeeeee;
|
||||||
|
}
|
||||||
|
.datepicker table tr td span.disabled,
|
||||||
|
.datepicker table tr td span.disabled:hover {
|
||||||
|
background: none;
|
||||||
|
color: #999999;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
.datepicker table tr td span.active,
|
||||||
|
.datepicker table tr td span.active:hover,
|
||||||
|
.datepicker table tr td span.active.disabled,
|
||||||
|
.datepicker table tr td span.active.disabled:hover {
|
||||||
|
background-color: #006dcc;
|
||||||
|
background-image: -moz-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: -ms-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#0088cc), to(#0044cc));
|
||||||
|
background-image: -webkit-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: -o-linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-image: linear-gradient(top, #0088cc, #0044cc);
|
||||||
|
background-repeat: repeat-x;
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#0088cc', endColorstr='#0044cc', GradientType=0);
|
||||||
|
border-color: #0044cc #0044cc #002a80;
|
||||||
|
border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
|
||||||
|
filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
|
||||||
|
color: #fff;
|
||||||
|
text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.25);
|
||||||
|
}
|
||||||
|
.datepicker table tr td span.active:hover,
|
||||||
|
.datepicker table tr td span.active:hover:hover,
|
||||||
|
.datepicker table tr td span.active.disabled:hover,
|
||||||
|
.datepicker table tr td span.active.disabled:hover:hover,
|
||||||
|
.datepicker table tr td span.active:active,
|
||||||
|
.datepicker table tr td span.active:hover:active,
|
||||||
|
.datepicker table tr td span.active.disabled:active,
|
||||||
|
.datepicker table tr td span.active.disabled:hover:active,
|
||||||
|
.datepicker table tr td span.active.active,
|
||||||
|
.datepicker table tr td span.active:hover.active,
|
||||||
|
.datepicker table tr td span.active.disabled.active,
|
||||||
|
.datepicker table tr td span.active.disabled:hover.active,
|
||||||
|
.datepicker table tr td span.active.disabled,
|
||||||
|
.datepicker table tr td span.active:hover.disabled,
|
||||||
|
.datepicker table tr td span.active.disabled.disabled,
|
||||||
|
.datepicker table tr td span.active.disabled:hover.disabled,
|
||||||
|
.datepicker table tr td span.active[disabled],
|
||||||
|
.datepicker table tr td span.active:hover[disabled],
|
||||||
|
.datepicker table tr td span.active.disabled[disabled],
|
||||||
|
.datepicker table tr td span.active.disabled:hover[disabled] {
|
||||||
|
background-color: #0044cc;
|
||||||
|
}
|
||||||
|
.datepicker table tr td span.active:active,
|
||||||
|
.datepicker table tr td span.active:hover:active,
|
||||||
|
.datepicker table tr td span.active.disabled:active,
|
||||||
|
.datepicker table tr td span.active.disabled:hover:active,
|
||||||
|
.datepicker table tr td span.active.active,
|
||||||
|
.datepicker table tr td span.active:hover.active,
|
||||||
|
.datepicker table tr td span.active.disabled.active,
|
||||||
|
.datepicker table tr td span.active.disabled:hover.active {
|
||||||
|
background-color: #003399 \9;
|
||||||
|
}
|
||||||
|
.datepicker table tr td span.old,
|
||||||
|
.datepicker table tr td span.new {
|
||||||
|
color: #999999;
|
||||||
|
}
|
||||||
|
.datepicker th.datepicker-switch {
|
||||||
|
width: 145px;
|
||||||
|
}
|
||||||
|
.datepicker thead tr:first-child th,
|
||||||
|
.datepicker tfoot tr th {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.datepicker thead tr:first-child th:hover,
|
||||||
|
.datepicker tfoot tr th:hover {
|
||||||
|
background: #eeeeee;
|
||||||
|
}
|
||||||
|
.datepicker .cw {
|
||||||
|
font-size: 10px;
|
||||||
|
width: 12px;
|
||||||
|
padding: 0 2px 0 5px;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.datepicker thead tr:first-child th.cw {
|
||||||
|
cursor: default;
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
.input-group.date .input-group-addon i,
|
||||||
|
.input-group.date .input-group-addon i {
|
||||||
|
display: block;
|
||||||
|
cursor: pointer;
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
}
|
||||||
|
.input-daterange input {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.input-daterange input:first-child {
|
||||||
|
-webkit-border-radius: 3px 0 0 3px;
|
||||||
|
-moz-border-radius: 3px 0 0 3px;
|
||||||
|
border-radius: 3px 0 0 3px;
|
||||||
|
}
|
||||||
|
.input-daterange input:last-child {
|
||||||
|
-webkit-border-radius: 0 3px 3px 0;
|
||||||
|
-moz-border-radius: 0 3px 3px 0;
|
||||||
|
border-radius: 0 3px 3px 0;
|
||||||
|
}
|
||||||
|
.input-daterange .add-on {
|
||||||
|
display: inline-block;
|
||||||
|
width: auto;
|
||||||
|
min-width: 16px;
|
||||||
|
height: 18px;
|
||||||
|
padding: 4px 5px;
|
||||||
|
font-weight: normal;
|
||||||
|
line-height: 18px;
|
||||||
|
text-align: center;
|
||||||
|
text-shadow: 0 1px 0 #ffffff;
|
||||||
|
vertical-align: middle;
|
||||||
|
background-color: #eeeeee;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
margin-left: -5px;
|
||||||
|
margin-right: -5px;
|
||||||
|
}
|
39
css/ie.css
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
.hideInIE8 {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-inner {
|
||||||
|
filter: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.box,
|
||||||
|
.discussions ul li {
|
||||||
|
border-top: none;
|
||||||
|
border-left: 1px solid #e4e6eb;
|
||||||
|
border-right: 1px solid #e4e6eb;
|
||||||
|
border-bottom: 2px solid #e4e6eb;
|
||||||
|
}
|
||||||
|
|
||||||
|
a.brand {
|
||||||
|
position: absolute;
|
||||||
|
left: 0px;
|
||||||
|
}
|
||||||
|
.container-fluid-full {
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.container-fluid-full #content {
|
||||||
|
width: 85.578%;
|
||||||
|
padding: 28px;
|
||||||
|
margin: 0px 0px;
|
||||||
|
margin-left: 14.422% !important;
|
||||||
|
}
|
||||||
|
.container-fluid-full #sidebar-left {
|
||||||
|
margin-left: 0px !important;
|
||||||
|
position: absolute;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
footer {
|
||||||
|
position: relative;
|
||||||
|
}
|
86
css/ie9.css
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
.navbar-inner {
|
||||||
|
filter: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dark {
|
||||||
|
right: -17px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.verticalChart .singleBar .bar .value span{
|
||||||
|
color: #3b3b41;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.box-header {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn3d:before {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.login-box {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderOverlay,
|
||||||
|
.progressBarOverlay {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.slider {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderVertical {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderBlue .ui-slider-range, .progressBlue .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderGreen .ui-slider-range, .progressGreen .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderDarkGreen .ui-slider-range, .progressDarkGreen .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderPink .ui-slider-range, .progressPink .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderOrange .ui-slider-range, .progressOrange .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderLightOrange .ui-slider-range, .progressLightOrange .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderRed .ui-slider-range, .progressRed .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.sliderYellow .ui-slider-range, .progressYellow .ui-progressbar-value {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.progressSlim {
|
||||||
|
background: url();
|
||||||
|
}
|
||||||
|
|
||||||
|
.quick-button,
|
||||||
|
.quick-button-small {
|
||||||
|
background: url();
|
||||||
|
|
||||||
|
}
|
459
css/style.css
Normal file
@ -0,0 +1,459 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
HumHub Style CSS
|
||||||
|
Author: Andreas Strobel
|
||||||
|
Website: http://www.humhub.org
|
||||||
|
|
||||||
|
|
||||||
|
Table of Bootstrap modifications:
|
||||||
|
---------------------------------------------
|
||||||
|
1) General
|
||||||
|
2) Topbar (replaces the Bootstrap Navbar)
|
||||||
|
3) Navs (Nav-Tabs & Nav-Pills)
|
||||||
|
4) Media Object
|
||||||
|
5) Panels
|
||||||
|
6) Pagination
|
||||||
|
7) Yii Components
|
||||||
|
8) HumHub Components
|
||||||
|
9) Third Party Tools
|
||||||
|
10) Responsive modifications
|
||||||
|
|
||||||
|
*/
|
||||||
|
body {
|
||||||
|
padding-top: 130px;
|
||||||
|
background-color: #f4f4f4;
|
||||||
|
}
|
||||||
|
a:hover {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
hr {
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
.col-md-1,
|
||||||
|
.col-md-2,
|
||||||
|
.col-md-3,
|
||||||
|
.col-md-4,
|
||||||
|
.col-md-5,
|
||||||
|
.col-md-6,
|
||||||
|
.col-md-7,
|
||||||
|
.col-md-8,
|
||||||
|
.col-md-9,
|
||||||
|
.col-md-10,
|
||||||
|
.col-md-11,
|
||||||
|
.col-md-12 {
|
||||||
|
position: inherit;
|
||||||
|
}
|
||||||
|
.topbar {
|
||||||
|
position: fixed;
|
||||||
|
display: block;
|
||||||
|
height: 50px;
|
||||||
|
width: 100%;
|
||||||
|
padding-left: 15px;
|
||||||
|
padding-right: 15px;
|
||||||
|
}
|
||||||
|
.topbar ul.nav {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
.topbar ul.nav > li {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
.topbar ul.nav > li > a {
|
||||||
|
padding-top: 15px;
|
||||||
|
padding-bottom: 15px;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
.topbar .dropdown-footer {
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
#topbar-first {
|
||||||
|
top: 0;
|
||||||
|
background-color: #101010;
|
||||||
|
z-index: 1030;
|
||||||
|
}
|
||||||
|
#topbar-first .nav > li > a:hover,
|
||||||
|
#topbar-first .nav > .open > a {
|
||||||
|
background-color: #2a2a2a;
|
||||||
|
}
|
||||||
|
#topbar-first .nav > .account {
|
||||||
|
height: 50px;
|
||||||
|
margin-left: 20px;
|
||||||
|
}
|
||||||
|
#topbar-first .nav > .account img {
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
#topbar-first .nav > .account .dropdown-toggle {
|
||||||
|
padding: 10px 5px 8px 5px;
|
||||||
|
line-height: 1.1em;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
#topbar-first .nav > .account .dropdown-toggle span {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
#topbar-first .topbar-brand {
|
||||||
|
position: relative;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
#topbar-first .topbar-actions {
|
||||||
|
position: relative;
|
||||||
|
z-index: 3;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
text-align: center;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications .btn-group {
|
||||||
|
position: relative;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications .btn-group > a {
|
||||||
|
padding: 5px 10px;
|
||||||
|
margin: 10px 2px;
|
||||||
|
display: inline-block;
|
||||||
|
border-radius: 2px;
|
||||||
|
text-decoration: none;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications .btn-group > .label {
|
||||||
|
position: absolute;
|
||||||
|
top: 4px;
|
||||||
|
right: -2px;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications {
|
||||||
|
width: 350px;
|
||||||
|
margin-left: -148px;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications li {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications li i.approval {
|
||||||
|
position: absolute;
|
||||||
|
left: 6px;
|
||||||
|
top: 35px;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications li i.accepted {
|
||||||
|
color: #5cb85c;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications li i.declined {
|
||||||
|
color: #d9534f;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications li .media {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
#topbar-first .notifications #dropdown-notifications li .media .img-space {
|
||||||
|
position: absolute;
|
||||||
|
top: 14px;
|
||||||
|
left: 14px;
|
||||||
|
}
|
||||||
|
#topbar-second {
|
||||||
|
top: 50px;
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
z-index: 1029;
|
||||||
|
}
|
||||||
|
#topbar-second .dropdown-menu {
|
||||||
|
padding-top: 0;
|
||||||
|
padding-bottom: 0;
|
||||||
|
}
|
||||||
|
#topbar-second #space-menu-dropdown,
|
||||||
|
#topbar-second #search-menu-dropdown {
|
||||||
|
width: 400px;
|
||||||
|
}
|
||||||
|
#topbar-second #space-menu-dropdown .media-list,
|
||||||
|
#topbar-second #search-menu-dropdown .media-list {
|
||||||
|
max-height: 400px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
#topbar-second #space-menu-dropdown form,
|
||||||
|
#topbar-second #search-menu-dropdown form {
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
#topbar-second #space-menu-dropdown .search-reset,
|
||||||
|
#topbar-second #search-menu-dropdown .search-reset {
|
||||||
|
position: absolute;
|
||||||
|
color: #999;
|
||||||
|
margin: 10px;
|
||||||
|
top: 8px;
|
||||||
|
right: 10px;
|
||||||
|
display: none;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.nav-tabs {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
.media-list li {
|
||||||
|
padding: 10px;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
.media-list li:hover,
|
||||||
|
.media-list li.selected {
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
.panel .statistics {
|
||||||
|
margin-top: 40px;
|
||||||
|
margin-right: 20px;
|
||||||
|
}
|
||||||
|
.panel .statistics .entry {
|
||||||
|
margin-left: 20px;
|
||||||
|
}
|
||||||
|
.panel .statistics .entry .count {
|
||||||
|
font-size: 30px;
|
||||||
|
}
|
||||||
|
.panel.profile {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
.panel.profile .controls {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: 10px;
|
||||||
|
}
|
||||||
|
.panel.members .panel-body a img,
|
||||||
|
.panel.groups .panel-body a img,
|
||||||
|
.panel.follower .panel-body a img,
|
||||||
|
.panel.spaces .panel-body a img {
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
.pagination-container {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.grid-view img {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
}
|
||||||
|
.errorMessage {
|
||||||
|
color: red;
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
.error {
|
||||||
|
border-color: red !important;
|
||||||
|
}
|
||||||
|
.loader {
|
||||||
|
height: 60px;
|
||||||
|
background: url('../img/loader.gif') no-repeat center center;
|
||||||
|
}
|
||||||
|
.loader-small {
|
||||||
|
height: 20px;
|
||||||
|
width: 20px;
|
||||||
|
}
|
||||||
|
ul.tag_input {
|
||||||
|
list-style: none;
|
||||||
|
background-color: #ffffff;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-radius: 4px;
|
||||||
|
-webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075);
|
||||||
|
box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075);
|
||||||
|
-webkit-transition: border-color ease-in-out 0.15s, box-shadow ease-in-out 0.15s;
|
||||||
|
transition: border-color ease-in-out 0.15s, box-shadow ease-in-out 0.15s;
|
||||||
|
padding: 0 0 9px 4px;
|
||||||
|
}
|
||||||
|
ul.tag_input li img {
|
||||||
|
margin: 0 5px 0 0;
|
||||||
|
}
|
||||||
|
.tag_input_field {
|
||||||
|
outline: none;
|
||||||
|
border: none !important;
|
||||||
|
padding: 5px 4px 0px 4px !important;
|
||||||
|
width: 170px;
|
||||||
|
margin: 2px 0 0 0 !important;
|
||||||
|
}
|
||||||
|
.userInput,
|
||||||
|
.spaceInput {
|
||||||
|
background-color: #428bca;
|
||||||
|
color: #ffffff;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-size: 12px !important;
|
||||||
|
padding: 2px;
|
||||||
|
float: left;
|
||||||
|
margin: 4px 4px 0 0;
|
||||||
|
}
|
||||||
|
.userInput i,
|
||||||
|
.spaceInput i {
|
||||||
|
padding: 0px 6px;
|
||||||
|
font-size: 14px;
|
||||||
|
cursor: pointer;
|
||||||
|
line-height: 8px;
|
||||||
|
}
|
||||||
|
#userpicker li > a,
|
||||||
|
#spacepicker li > a {
|
||||||
|
padding-left: 10px;
|
||||||
|
}
|
||||||
|
#userpicker li.selected,
|
||||||
|
#spacepicker li.selected {
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
.activities {
|
||||||
|
max-height: 400px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
.activities li .media {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
.activities li .media .img-space {
|
||||||
|
position: absolute;
|
||||||
|
top: 14px;
|
||||||
|
left: 14px;
|
||||||
|
}
|
||||||
|
.wall-entry {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
.wall-entry .media {
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
|
.wall-entry .media .img-space {
|
||||||
|
position: absolute;
|
||||||
|
top: 60px;
|
||||||
|
left: 60px;
|
||||||
|
}
|
||||||
|
.wall-entry .well {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
.wall-entry .well .comment .show-all-link {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
.media:after {
|
||||||
|
content: '';
|
||||||
|
clear: both;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.wall-entry .preferences {
|
||||||
|
position: absolute;
|
||||||
|
right: 1px;
|
||||||
|
top: 1px;
|
||||||
|
}
|
||||||
|
.wall-entry-controls {
|
||||||
|
font-size: 12px;
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
.tags {
|
||||||
|
margin: 5px 0;
|
||||||
|
}
|
||||||
|
.tags .tag {
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
.placeholder {
|
||||||
|
padding: 15px;
|
||||||
|
}
|
||||||
|
.space-member-sign {
|
||||||
|
color: #5cb85c;
|
||||||
|
position: absolute;
|
||||||
|
top: 50px;
|
||||||
|
left: 50px;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
.mime {
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: 0 0;
|
||||||
|
}
|
||||||
|
.mime-word {
|
||||||
|
background-image: url('../img/mime/word.png');
|
||||||
|
}
|
||||||
|
.mime-excel {
|
||||||
|
background-image: url('../img/mime/excel.png');
|
||||||
|
}
|
||||||
|
.mime-powerpoint {
|
||||||
|
background-image: url('../img/mime/powerpoint.png');
|
||||||
|
}
|
||||||
|
.mime-pdf {
|
||||||
|
background-image: url('../img/mime/pdf.png');
|
||||||
|
}
|
||||||
|
.mime-zip {
|
||||||
|
background-image: url('../img/mime/zip.png');
|
||||||
|
}
|
||||||
|
.mime-image {
|
||||||
|
background-image: url('../img/mime/image.png');
|
||||||
|
}
|
||||||
|
.mime-file {
|
||||||
|
background-image: url('../img/mime/file.png');
|
||||||
|
}
|
||||||
|
.mime-photoshop {
|
||||||
|
background-image: url('../img/mime/photoshop.png');
|
||||||
|
}
|
||||||
|
.mime-illustrator {
|
||||||
|
background-image: url('../img/mime/illustrator.png');
|
||||||
|
}
|
||||||
|
.mime-flash {
|
||||||
|
background-image: url('../img/mime/flash.png');
|
||||||
|
}
|
||||||
|
.mime-audio {
|
||||||
|
background-image: url('../img/mime/audio.png');
|
||||||
|
}
|
||||||
|
.files {
|
||||||
|
padding-left: 0;
|
||||||
|
}
|
||||||
|
.highlight {
|
||||||
|
background-color: #fffbc1;
|
||||||
|
}
|
||||||
|
@media (max-width: 767px) {
|
||||||
|
.topbar {
|
||||||
|
padding-left: 0;
|
||||||
|
padding-right: 0;
|
||||||
|
}
|
||||||
|
#space-menu > .title {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
#space-menu-dropdown {
|
||||||
|
width: 300px !important;
|
||||||
|
}
|
||||||
|
#search-menu-dropdown {
|
||||||
|
width: 300px !important;
|
||||||
|
}
|
||||||
|
.notifications {
|
||||||
|
position: inherit !important;
|
||||||
|
float: left !important;
|
||||||
|
}
|
||||||
|
#dropdown-notifications {
|
||||||
|
width: 300px !important;
|
||||||
|
margin-left: 0 !important;
|
||||||
|
}
|
||||||
|
.profile .media > a {
|
||||||
|
float: none !important;
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
.profile .media > a img {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
.profile .statistics {
|
||||||
|
float: none !important;
|
||||||
|
}
|
||||||
|
/*.wall-entry {
|
||||||
|
|
||||||
|
.media > a {
|
||||||
|
float: none !important;
|
||||||
|
|
||||||
|
img {
|
||||||
|
display: inline;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.comment .media > a {
|
||||||
|
float: left !important;
|
||||||
|
|
||||||
|
img {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.media-body {
|
||||||
|
.media-heading {
|
||||||
|
position: absolute;
|
||||||
|
top: 20px;
|
||||||
|
left: 100px;
|
||||||
|
|
||||||
|
small {
|
||||||
|
display: block;
|
||||||
|
padding-top: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}*/
|
||||||
|
}
|
620
css/style.less
Normal file
@ -0,0 +1,620 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
HumHub Style CSS
|
||||||
|
Author: Andreas Strobel
|
||||||
|
Website: http://www.humhub.org
|
||||||
|
|
||||||
|
|
||||||
|
Table of Bootstrap modifications:
|
||||||
|
---------------------------------------------
|
||||||
|
1) General
|
||||||
|
2) Topbar (replaces the Bootstrap Navbar)
|
||||||
|
3) Navs (Nav-Tabs & Nav-Pills)
|
||||||
|
4) Media Object
|
||||||
|
5) Panels
|
||||||
|
6) Pagination
|
||||||
|
7) Yii Components
|
||||||
|
8) HumHub Components
|
||||||
|
9) Third Party Tools
|
||||||
|
10) Responsive modifications
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
//
|
||||||
|
// 1) General
|
||||||
|
// --------------------------------------------------
|
||||||
|
|
||||||
|
body {
|
||||||
|
padding-top: 130px;
|
||||||
|
background-color: #f4f4f4;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cols (change position property to prevent the cutting of tooltips or menus)
|
||||||
|
.col-md-1,
|
||||||
|
.col-md-2,
|
||||||
|
.col-md-3,
|
||||||
|
.col-md-4,
|
||||||
|
.col-md-5,
|
||||||
|
.col-md-6,
|
||||||
|
.col-md-7,
|
||||||
|
.col-md-8,
|
||||||
|
.col-md-9,
|
||||||
|
.col-md-10,
|
||||||
|
.col-md-11,
|
||||||
|
.col-md-12 {
|
||||||
|
position: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 2) Topbar
|
||||||
|
// --------------------------------------------------
|
||||||
|
.topbar {
|
||||||
|
position: fixed;
|
||||||
|
display: block;
|
||||||
|
height: 50px;
|
||||||
|
width: 100%;
|
||||||
|
padding-left: 15px;
|
||||||
|
padding-right: 15px;
|
||||||
|
|
||||||
|
ul.nav {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.nav > li {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.nav > li > a {
|
||||||
|
padding-top: 15px;
|
||||||
|
padding-bottom: 15px;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-footer {
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#topbar-first {
|
||||||
|
top: 0;
|
||||||
|
background-color: #101010;
|
||||||
|
z-index: 1030;
|
||||||
|
|
||||||
|
.nav > li > a:hover, .nav > .open > a {
|
||||||
|
background-color: lighten(#101010, 10%);;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav > .account {
|
||||||
|
height: 50px;
|
||||||
|
margin-left: 20px;
|
||||||
|
|
||||||
|
img {
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-toggle {
|
||||||
|
padding: 10px 5px 8px 5px;
|
||||||
|
line-height: 1.1em;
|
||||||
|
text-align: left;
|
||||||
|
|
||||||
|
span {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.topbar-brand {
|
||||||
|
position: relative;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topbar-actions {
|
||||||
|
position: relative;
|
||||||
|
z-index: 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.notifications {
|
||||||
|
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
text-align: center;
|
||||||
|
z-index: 1;
|
||||||
|
|
||||||
|
.btn-group {
|
||||||
|
position: relative;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-group > a {
|
||||||
|
padding: 5px 10px;
|
||||||
|
margin: 10px 2px;
|
||||||
|
display: inline-block;
|
||||||
|
border-radius: 2px;
|
||||||
|
text-decoration: none;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-group > .label {
|
||||||
|
position: absolute;
|
||||||
|
top: 4px;
|
||||||
|
right: -2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dropdown-notifications {
|
||||||
|
width: 350px;
|
||||||
|
margin-left: -148px;
|
||||||
|
|
||||||
|
li {
|
||||||
|
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
i.approval {
|
||||||
|
position: absolute;
|
||||||
|
left: 6px;
|
||||||
|
top: 35px;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
i.accepted {
|
||||||
|
color: #5cb85c;
|
||||||
|
}
|
||||||
|
|
||||||
|
i.declined {
|
||||||
|
color: #d9534f;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
li .media {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
li .media .img-space {
|
||||||
|
position: absolute;
|
||||||
|
top: 14px;
|
||||||
|
left: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#topbar-second {
|
||||||
|
top: 50px;
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
z-index: 1029;
|
||||||
|
|
||||||
|
.dropdown-menu {
|
||||||
|
padding-top: 0;
|
||||||
|
padding-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Space dropdown menu
|
||||||
|
#space-menu-dropdown, #search-menu-dropdown {
|
||||||
|
|
||||||
|
width: 400px;
|
||||||
|
|
||||||
|
.media-list {
|
||||||
|
max-height: 400px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
form {
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-reset {
|
||||||
|
position: absolute;
|
||||||
|
color: #999;
|
||||||
|
margin: 10px;
|
||||||
|
top: 8px;
|
||||||
|
right: 10px;
|
||||||
|
display: none;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 3) Navs (Nav-Tabs & Nav-Pills)
|
||||||
|
// --------------------------------------------------
|
||||||
|
.nav-tabs {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 4) Media-Object
|
||||||
|
// --------------------------------------------------
|
||||||
|
.media-list {
|
||||||
|
li {
|
||||||
|
padding: 10px;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
li:hover, li.selected {
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 5) Panels
|
||||||
|
// --------------------------------------------------
|
||||||
|
|
||||||
|
.panel {
|
||||||
|
|
||||||
|
.statistics {
|
||||||
|
|
||||||
|
margin-top: 40px;
|
||||||
|
margin-right: 20px;
|
||||||
|
|
||||||
|
.entry {
|
||||||
|
margin-left: 20px;
|
||||||
|
|
||||||
|
.count {
|
||||||
|
font-size: 30px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel.profile {
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
.controls {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: 10px;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel.members, .panel.groups, .panel.follower, .panel.spaces {
|
||||||
|
.panel-body a img {
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 6) Pagination
|
||||||
|
// --------------------------------------------------
|
||||||
|
.pagination-container {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 7) Yii Components
|
||||||
|
// --------------------------------------------------
|
||||||
|
|
||||||
|
.grid-view {
|
||||||
|
img {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Errror handling
|
||||||
|
.errorMessage {
|
||||||
|
color: red;
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error {
|
||||||
|
border-color: red !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 8) HumHub Components
|
||||||
|
// --------------------------------------------------
|
||||||
|
|
||||||
|
// Ajax loader
|
||||||
|
.loader {
|
||||||
|
height: 60px;
|
||||||
|
background: url('../img/loader.gif') no-repeat center center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader-small {
|
||||||
|
height: 20px;
|
||||||
|
width: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
// User- & Space picker
|
||||||
|
ul.tag_input {
|
||||||
|
list-style: none;
|
||||||
|
background-color: #ffffff;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-radius: 4px;
|
||||||
|
-webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075);
|
||||||
|
box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075);
|
||||||
|
-webkit-transition: border-color ease-in-out .15s, box-shadow ease-in-out .15s;
|
||||||
|
transition: border-color ease-in-out .15s, box-shadow ease-in-out .15s;
|
||||||
|
|
||||||
|
padding: 0 0 9px 4px;
|
||||||
|
|
||||||
|
li img {
|
||||||
|
margin: 0 5px 0 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.tag_input_field {
|
||||||
|
outline: none;
|
||||||
|
border: none !important;
|
||||||
|
padding: 5px 4px 0px 4px !important;
|
||||||
|
width: 170px;
|
||||||
|
margin: 2px 0 0 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.userInput, .spaceInput {
|
||||||
|
background-color: #428bca;
|
||||||
|
color: #ffffff;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-size: 12px !important;
|
||||||
|
padding: 2px;
|
||||||
|
float: left;
|
||||||
|
margin: 4px 4px 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.userInput i, .spaceInput i {
|
||||||
|
padding: 0px 6px;
|
||||||
|
font-size: 14px;
|
||||||
|
cursor: pointer;
|
||||||
|
line-height: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#userpicker, #spacepicker {
|
||||||
|
li > a {
|
||||||
|
padding-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
li.selected {
|
||||||
|
//padding-left: 6px;
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Activities
|
||||||
|
.activities {
|
||||||
|
|
||||||
|
max-height: 400px;
|
||||||
|
overflow: auto;
|
||||||
|
|
||||||
|
li .media {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
li .media .img-space {
|
||||||
|
position: absolute;
|
||||||
|
top: 14px;
|
||||||
|
left: 14px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wall-Entries
|
||||||
|
.wall-entry {
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
.media {
|
||||||
|
overflow: visible;
|
||||||
|
|
||||||
|
.img-space {
|
||||||
|
position: absolute;
|
||||||
|
top: 60px;
|
||||||
|
left: 60px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.well {
|
||||||
|
margin-bottom: 0;
|
||||||
|
|
||||||
|
.comment {
|
||||||
|
.show-all-link {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.media:after {
|
||||||
|
content: '';
|
||||||
|
clear: both;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wall-entry {
|
||||||
|
.preferences {
|
||||||
|
position: absolute;
|
||||||
|
right: 1px;
|
||||||
|
top: 1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.wall-entry-controls {
|
||||||
|
font-size: 12px;
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tags
|
||||||
|
.tags {
|
||||||
|
|
||||||
|
margin: 5px 0;
|
||||||
|
|
||||||
|
.tag {
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Placeholder text for empty content
|
||||||
|
.placeholder {
|
||||||
|
padding: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Member sign for directory list
|
||||||
|
.space-member-sign {
|
||||||
|
color: #5cb85c;
|
||||||
|
position: absolute;
|
||||||
|
top: 50px;
|
||||||
|
left: 50px;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Mime-Types
|
||||||
|
.mime {
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: 0 0;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-word {
|
||||||
|
background-image: url('../img/mime/word.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-excel {
|
||||||
|
background-image: url('../img/mime/excel.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-powerpoint {
|
||||||
|
background-image: url('../img/mime/powerpoint.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-pdf {
|
||||||
|
background-image: url('../img/mime/pdf.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-zip {
|
||||||
|
background-image: url('../img/mime/zip.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-image {
|
||||||
|
background-image: url('../img/mime/image.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-file {
|
||||||
|
background-image: url('../img/mime/file.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-photoshop {
|
||||||
|
background-image: url('../img/mime/photoshop.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-illustrator {
|
||||||
|
background-image: url('../img/mime/illustrator.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-flash {
|
||||||
|
background-image: url('../img/mime/flash.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.mime-audio {
|
||||||
|
background-image: url('../img/mime/audio.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
.files {
|
||||||
|
padding-left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 9) Third Party Tools
|
||||||
|
// --------------------------------------------------
|
||||||
|
.highlight {
|
||||||
|
background-color: #fffbc1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// 10) Responsive modifications
|
||||||
|
// --------------------------------------------------
|
||||||
|
@media (max-width: 767px) {
|
||||||
|
|
||||||
|
.topbar {
|
||||||
|
padding-left: 0;
|
||||||
|
padding-right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#space-menu > .title {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#space-menu-dropdown {
|
||||||
|
width: 300px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
#search-menu-dropdown {
|
||||||
|
width: 300px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.notifications {
|
||||||
|
position: inherit !important;
|
||||||
|
float: left !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dropdown-notifications {
|
||||||
|
width: 300px !important;
|
||||||
|
margin-left: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.profile {
|
||||||
|
|
||||||
|
.media > a {
|
||||||
|
float: none !important;
|
||||||
|
margin-right: 0;
|
||||||
|
|
||||||
|
img {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.statistics {
|
||||||
|
float: none !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*.wall-entry {
|
||||||
|
|
||||||
|
.media > a {
|
||||||
|
float: none !important;
|
||||||
|
|
||||||
|
img {
|
||||||
|
display: inline;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.comment .media > a {
|
||||||
|
float: left !important;
|
||||||
|
|
||||||
|
img {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.media-body {
|
||||||
|
.media-heading {
|
||||||
|
position: absolute;
|
||||||
|
top: 20px;
|
||||||
|
left: 100px;
|
||||||
|
|
||||||
|
small {
|
||||||
|
display: block;
|
||||||
|
padding-top: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}*/
|
||||||
|
}
|
BIN
ico/apple-touch-icon-114-precomposed.png
Normal file
After Width: | Height: | Size: 1.8 KiB |
BIN
ico/apple-touch-icon-144-precomposed.png
Normal file
After Width: | Height: | Size: 1.9 KiB |
BIN
ico/apple-touch-icon-57-precomposed.png
Normal file
After Width: | Height: | Size: 1.4 KiB |
BIN
ico/apple-touch-icon-72-precomposed.png
Normal file
After Width: | Height: | Size: 1.5 KiB |
BIN
ico/favicon.ico
Normal file
After Width: | Height: | Size: 1.1 KiB |
BIN
img/glyphicons-halflings-white.png
Normal file
After Width: | Height: | Size: 8.6 KiB |
BIN
img/glyphicons-halflings.png
Normal file
After Width: | Height: | Size: 12 KiB |
BIN
img/loader.gif
Normal file
After Width: | Height: | Size: 673 B |
BIN
img/mime/audio.png
Normal file
After Width: | Height: | Size: 635 B |
BIN
img/mime/excel.png
Normal file
After Width: | Height: | Size: 581 B |
BIN
img/mime/file.png
Normal file
After Width: | Height: | Size: 352 B |
BIN
img/mime/flash.png
Normal file
After Width: | Height: | Size: 544 B |
BIN
img/mime/illustrator.png
Normal file
After Width: | Height: | Size: 507 B |
BIN
img/mime/image.png
Normal file
After Width: | Height: | Size: 528 B |
BIN
img/mime/pdf.png
Normal file
After Width: | Height: | Size: 500 B |
BIN
img/mime/photoshop.png
Normal file
After Width: | Height: | Size: 535 B |
BIN
img/mime/powerpoint.png
Normal file
After Width: | Height: | Size: 449 B |
BIN
img/mime/word.png
Normal file
After Width: | Height: | Size: 535 B |
BIN
img/mime/zip.png
Normal file
After Width: | Height: | Size: 475 B |
39
index.php
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
$yii = dirname(__FILE__) . '/protected/vendors/yii/yii.php';
|
||||||
|
$config = dirname(__FILE__) . '/protected/config/main.php';
|
||||||
|
$appClass = dirname(__FILE__) . '/protected/components/WebApplication.php';
|
||||||
|
|
||||||
|
// enable the following 3 lines when in development/testing
|
||||||
|
defined('YII_DEBUG') or define('YII_DEBUG', true);
|
||||||
|
defined('YII_TRACE_LEVEL') or define('YII_TRACE_LEVEL', 3);
|
||||||
|
ini_set('error_reporting', E_ALL);
|
||||||
|
|
||||||
|
require_once($yii);
|
||||||
|
require_once($appClass);
|
||||||
|
|
||||||
|
$app = Yii::createApplication('WebApplication', $config);
|
||||||
|
|
||||||
|
Yii::import('application.vendors.*');
|
||||||
|
EZendAutoloader::$prefixes = array('Zend', 'Custom');
|
||||||
|
Yii::import("ext.yiiext.components.zendAutoloader.EZendAutoloader", true);
|
||||||
|
Yii::registerAutoloader(array("EZendAutoloader", "loadClass"), true);
|
||||||
|
|
||||||
|
$app->run();
|
132
js/app.js
Normal file
@ -0,0 +1,132 @@
|
|||||||
|
/**
|
||||||
|
* Holds all already loaded javascript libaries
|
||||||
|
*
|
||||||
|
* @type HashTable
|
||||||
|
*/
|
||||||
|
var currentLoadedJavaScripts = new HashTable();
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Looks for script tags inside the given string and checks if the files
|
||||||
|
* are already loaded.
|
||||||
|
*
|
||||||
|
* When already loaded, the scripts will ignored.
|
||||||
|
*
|
||||||
|
* @returns {undefined}
|
||||||
|
*/
|
||||||
|
function parseHtml(htmlString) {
|
||||||
|
var re = /<script type="text\/javascript" src="([\s\S]*?)"><\/script>/gm;
|
||||||
|
|
||||||
|
var match;
|
||||||
|
while (match = re.exec(htmlString)) {
|
||||||
|
|
||||||
|
js = match[1];
|
||||||
|
|
||||||
|
if (currentLoadedJavaScripts.hasItem(js)) {
|
||||||
|
// Remove Script Tag
|
||||||
|
//console.log("Ignore load of : "+js);
|
||||||
|
htmlString = htmlString.replace(match[0], "");
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// Let Script Tag
|
||||||
|
//console.log("First load of: "+js);
|
||||||
|
currentLoadedJavaScripts.setItem(js, 1);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
return htmlString;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hashtable
|
||||||
|
*
|
||||||
|
* Javscript Class which represents a hashtable.
|
||||||
|
*
|
||||||
|
* @param {type} obj
|
||||||
|
* @returns {HashTable}
|
||||||
|
*/
|
||||||
|
function HashTable(obj)
|
||||||
|
{
|
||||||
|
this.length = 0;
|
||||||
|
this.items = {};
|
||||||
|
for (var p in obj) {
|
||||||
|
if (obj.hasOwnProperty(p)) {
|
||||||
|
this.items[p] = obj[p];
|
||||||
|
this.length++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.setItem = function(key, value)
|
||||||
|
{
|
||||||
|
var previous = undefined;
|
||||||
|
if (this.hasItem(key)) {
|
||||||
|
previous = this.items[key];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.length++;
|
||||||
|
}
|
||||||
|
this.items[key] = value;
|
||||||
|
return previous;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.getItem = function(key) {
|
||||||
|
return this.hasItem(key) ? this.items[key] : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.hasItem = function(key)
|
||||||
|
{
|
||||||
|
return this.items.hasOwnProperty(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.removeItem = function(key)
|
||||||
|
{
|
||||||
|
if (this.hasItem(key)) {
|
||||||
|
previous = this.items[key];
|
||||||
|
this.length--;
|
||||||
|
delete this.items[key];
|
||||||
|
return previous;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.keys = function()
|
||||||
|
{
|
||||||
|
var keys = [];
|
||||||
|
for (var k in this.items) {
|
||||||
|
if (this.hasItem(k)) {
|
||||||
|
keys.push(k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keys;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.values = function()
|
||||||
|
{
|
||||||
|
var values = [];
|
||||||
|
for (var k in this.items) {
|
||||||
|
if (this.hasItem(k)) {
|
||||||
|
values.push(this.items[k]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return values;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.each = function(fn) {
|
||||||
|
for (var k in this.items) {
|
||||||
|
if (this.hasItem(k)) {
|
||||||
|
fn(k, this.items[k]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.clear = function()
|
||||||
|
{
|
||||||
|
this.items = {}
|
||||||
|
this.length = 0;
|
||||||
|
}
|
||||||
|
}
|
1396
js/bootstrap-datepicker.js
vendored
Normal file
9
js/bootstrap.min.js
vendored
Normal file
521
js/bootstrap3-wysihtml5.js
vendored
Normal file
@ -0,0 +1,521 @@
|
|||||||
|
!function($, wysi) {
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var tpl = {
|
||||||
|
"font-styles": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li class='dropdown'>" +
|
||||||
|
"<a class='btn dropdown-toggle btn-" + size + " btn-default' data-toggle='dropdown' href='#'>" +
|
||||||
|
"<i class='glyphicon glyphicon-font'></i> <span class='current-font'>" + locale.font_styles.normal + "</span> <b class='caret'></b>" +
|
||||||
|
"</a>" +
|
||||||
|
"<ul class='dropdown-menu'>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='div' tabindex='-1'>" + locale.font_styles.normal + "</a></li>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='h1' tabindex='-1'>" + locale.font_styles.h1 + "</a></li>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='h2' tabindex='-1'>" + locale.font_styles.h2 + "</a></li>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='h3' tabindex='-1'>" + locale.font_styles.h3 + "</a></li>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='h4'>" + locale.font_styles.h4 + "</a></li>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='h5'>" + locale.font_styles.h5 + "</a></li>" +
|
||||||
|
"<li><a data-wysihtml5-command='formatBlock' data-wysihtml5-command-value='h6'>" + locale.font_styles.h6 + "</a></li>" +
|
||||||
|
"</ul>" +
|
||||||
|
"</li>";
|
||||||
|
},
|
||||||
|
|
||||||
|
"emphasis": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li>" +
|
||||||
|
"<div class='btn-group'>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='bold' title='CTRL+B' tabindex='-1'>" + locale.emphasis.bold + "</a>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='italic' title='CTRL+I' tabindex='-1'>" + locale.emphasis.italic + "</a>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='underline' title='CTRL+U' tabindex='-1'>" + locale.emphasis.underline + "</a>" +
|
||||||
|
"</div>" +
|
||||||
|
"</li>";
|
||||||
|
},
|
||||||
|
|
||||||
|
"lists": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li>" +
|
||||||
|
"<div class='btn-group'>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='insertUnorderedList' title='" + locale.lists.unordered + "' tabindex='-1'><i class='glyphicon glyphicon-list'></i></a>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='insertOrderedList' title='" + locale.lists.ordered + "' tabindex='-1'><i class='glyphicon glyphicon-th-list'></i></a>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='Outdent' title='" + locale.lists.outdent + "' tabindex='-1'><i class='glyphicon glyphicon-indent-right'></i></a>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='Indent' title='" + locale.lists.indent + "' tabindex='-1'><i class='glyphicon glyphicon-indent-left'></i></a>" +
|
||||||
|
"</div>" +
|
||||||
|
"</li>";
|
||||||
|
},
|
||||||
|
|
||||||
|
"link": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li>" +
|
||||||
|
""+
|
||||||
|
"<div class='bootstrap-wysihtml5-insert-link-modal modal fade'>" +
|
||||||
|
"<div class='modal-dialog'>"+
|
||||||
|
"<div class='modal-content'>"+
|
||||||
|
"<div class='modal-header'>" +
|
||||||
|
"<a class='close' data-dismiss='modal'>×</a>" +
|
||||||
|
"<h4>" + locale.link.insert + "</h4>" +
|
||||||
|
"</div>" +
|
||||||
|
"<div class='modal-body'>" +
|
||||||
|
"<input value='http://' class='bootstrap-wysihtml5-insert-link-url form-control'>" +
|
||||||
|
"<label class='checkbox'> <input type='checkbox' class='bootstrap-wysihtml5-insert-link-target' checked>" + locale.link.target + "</label>" +
|
||||||
|
"</div>" +
|
||||||
|
"<div class='modal-footer'>" +
|
||||||
|
"<button class='btn btn-default' data-dismiss='modal'>" + locale.link.cancel + "</button>" +
|
||||||
|
"<button href='#' class='btn btn-primary' data-dismiss='modal'>" + locale.link.insert + "</button>" +
|
||||||
|
"</div>" +
|
||||||
|
"</div>" +
|
||||||
|
"</div>" +
|
||||||
|
"</div>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='createLink' title='" + locale.link.insert + "' tabindex='-1'><i class='glyphicon glyphicon-share'></i></a>" +
|
||||||
|
"</li>";
|
||||||
|
},
|
||||||
|
|
||||||
|
"image": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li>" +
|
||||||
|
"<div class='bootstrap-wysihtml5-insert-image-modal modal fade'>" +
|
||||||
|
"<div class='modal-dialog'>"+
|
||||||
|
"<div class='modal-content'>"+
|
||||||
|
"<div class='modal-header'>" +
|
||||||
|
"<a class='close' data-dismiss='modal'>×</a>" +
|
||||||
|
"<h4>" + locale.image.insert + "</h4>" +
|
||||||
|
"</div>" +
|
||||||
|
"<div class='modal-body'>" +
|
||||||
|
"<input value='http://' class='bootstrap-wysihtml5-insert-image-url form-control'>" +
|
||||||
|
"</div>" +
|
||||||
|
"<div class='modal-footer'>" +
|
||||||
|
"<button class='btn btn-default' data-dismiss='modal'>" + locale.image.cancel + "</button>" +
|
||||||
|
"<button class='btn btn-primary' data-dismiss='modal'>" + locale.image.insert + "</button>" +
|
||||||
|
"</div>" +
|
||||||
|
"</div>" +
|
||||||
|
"</div>" +
|
||||||
|
"</div>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-command='insertImage' title='" + locale.image.insert + "' tabindex='-1'><i class='glyphicon glyphicon-picture'></i></a>" +
|
||||||
|
"</li>";
|
||||||
|
},
|
||||||
|
|
||||||
|
"html": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li>" +
|
||||||
|
"<div class='btn-group'>" +
|
||||||
|
"<a class='btn btn-" + size + " btn-default' data-wysihtml5-action='change_view' title='" + locale.html.edit + "' tabindex='-1'><i class='glyphicon glyphicon-pencil'></i></a>" +
|
||||||
|
"</div>" +
|
||||||
|
"</li>";
|
||||||
|
},
|
||||||
|
|
||||||
|
"color": function(locale, options) {
|
||||||
|
var size = (options && options.size) ? ' btn-'+options.size : '';
|
||||||
|
return "<li class='dropdown'>" +
|
||||||
|
"<a class='btn dropdown-toggle btn-" + size + " btn-default' data-toggle='dropdown' href='#' tabindex='-1'>" +
|
||||||
|
"<span class='current-color'>" + locale.colours.black + "</span> <b class='caret'></b>" +
|
||||||
|
"</a>" +
|
||||||
|
"<ul class='dropdown-menu'>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='black'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='black'>" + locale.colours.black + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='silver'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='silver'>" + locale.colours.silver + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='gray'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='gray'>" + locale.colours.gray + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='maroon'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='maroon'>" + locale.colours.maroon + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='red'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='red'>" + locale.colours.red + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='purple'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='purple'>" + locale.colours.purple + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='green'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='green'>" + locale.colours.green + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='olive'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='olive'>" + locale.colours.olive + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='navy'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='navy'>" + locale.colours.navy + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='blue'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='blue'>" + locale.colours.blue + "</a></li>" +
|
||||||
|
"<li><div class='wysihtml5-colors' data-wysihtml5-command-value='orange'></div><a class='wysihtml5-colors-title' data-wysihtml5-command='foreColor' data-wysihtml5-command-value='orange'>" + locale.colours.orange + "</a></li>" +
|
||||||
|
"</ul>" +
|
||||||
|
"</li>";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var templates = function(key, locale, options) {
|
||||||
|
return tpl[key](locale, options);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
var Wysihtml5 = function(el, options) {
|
||||||
|
this.el = el;
|
||||||
|
var toolbarOpts = options || defaultOptions;
|
||||||
|
for(var t in toolbarOpts.customTemplates) {
|
||||||
|
tpl[t] = toolbarOpts.customTemplates[t];
|
||||||
|
}
|
||||||
|
this.toolbar = this.createToolbar(el, toolbarOpts);
|
||||||
|
this.editor = this.createEditor(options);
|
||||||
|
|
||||||
|
window.editor = this.editor;
|
||||||
|
|
||||||
|
$('iframe.wysihtml5-sandbox').each(function(i, el){
|
||||||
|
$(el.contentWindow).off('focus.wysihtml5').on({
|
||||||
|
'focus.wysihtml5' : function(){
|
||||||
|
$('li.dropdown').removeClass('open');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
Wysihtml5.prototype = {
|
||||||
|
|
||||||
|
constructor: Wysihtml5,
|
||||||
|
|
||||||
|
createEditor: function(options) {
|
||||||
|
options = options || {};
|
||||||
|
|
||||||
|
// Add the toolbar to a clone of the options object so multiple instances
|
||||||
|
// of the WYISYWG don't break because "toolbar" is already defined
|
||||||
|
options = $.extend(true, {}, options);
|
||||||
|
options.toolbar = this.toolbar[0];
|
||||||
|
|
||||||
|
var editor = new wysi.Editor(this.el[0], options);
|
||||||
|
|
||||||
|
if(options && options.events) {
|
||||||
|
for(var eventName in options.events) {
|
||||||
|
editor.on(eventName, options.events[eventName]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return editor;
|
||||||
|
},
|
||||||
|
|
||||||
|
createToolbar: function(el, options) {
|
||||||
|
var self = this;
|
||||||
|
var toolbar = $("<ul/>", {
|
||||||
|
'class' : "wysihtml5-toolbar",
|
||||||
|
'style': "display:none"
|
||||||
|
});
|
||||||
|
var culture = options.locale || defaultOptions.locale || "en";
|
||||||
|
for(var key in defaultOptions) {
|
||||||
|
var value = false;
|
||||||
|
|
||||||
|
if(options[key] !== undefined) {
|
||||||
|
if(options[key] === true) {
|
||||||
|
value = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
value = defaultOptions[key];
|
||||||
|
}
|
||||||
|
|
||||||
|
if(value === true) {
|
||||||
|
toolbar.append(templates(key, locale[culture], options));
|
||||||
|
|
||||||
|
if(key === "html") {
|
||||||
|
this.initHtml(toolbar);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(key === "link") {
|
||||||
|
this.initInsertLink(toolbar);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(key === "image") {
|
||||||
|
this.initInsertImage(toolbar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(options.toolbar) {
|
||||||
|
for(key in options.toolbar) {
|
||||||
|
toolbar.append(options.toolbar[key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
toolbar.find("a[data-wysihtml5-command='formatBlock']").click(function(e) {
|
||||||
|
var target = e.target || e.srcElement;
|
||||||
|
var el = $(target);
|
||||||
|
self.toolbar.find('.current-font').text(el.html());
|
||||||
|
});
|
||||||
|
|
||||||
|
toolbar.find("a[data-wysihtml5-command='foreColor']").click(function(e) {
|
||||||
|
var target = e.target || e.srcElement;
|
||||||
|
var el = $(target);
|
||||||
|
self.toolbar.find('.current-color').text(el.html());
|
||||||
|
});
|
||||||
|
|
||||||
|
this.el.before(toolbar);
|
||||||
|
|
||||||
|
return toolbar;
|
||||||
|
},
|
||||||
|
|
||||||
|
initHtml: function(toolbar) {
|
||||||
|
var changeViewSelector = "a[data-wysihtml5-action='change_view']";
|
||||||
|
toolbar.find(changeViewSelector).click(function(e) {
|
||||||
|
toolbar.find('a.btn').not(changeViewSelector).toggleClass('disabled');
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
initInsertImage: function(toolbar) {
|
||||||
|
var self = this;
|
||||||
|
var insertImageModal = toolbar.find('.bootstrap-wysihtml5-insert-image-modal');
|
||||||
|
var urlInput = insertImageModal.find('.bootstrap-wysihtml5-insert-image-url');
|
||||||
|
var insertButton = insertImageModal.find('.btn-primary');
|
||||||
|
var initialValue = urlInput.val();
|
||||||
|
var caretBookmark;
|
||||||
|
|
||||||
|
var insertImage = function() {
|
||||||
|
var url = urlInput.val();
|
||||||
|
urlInput.val(initialValue);
|
||||||
|
self.editor.currentView.element.focus();
|
||||||
|
if (caretBookmark) {
|
||||||
|
self.editor.composer.selection.setBookmark(caretBookmark);
|
||||||
|
caretBookmark = null;
|
||||||
|
}
|
||||||
|
self.editor.composer.commands.exec("insertImage", url);
|
||||||
|
};
|
||||||
|
|
||||||
|
urlInput.keypress(function(e) {
|
||||||
|
if(e.which == 13) {
|
||||||
|
insertImage();
|
||||||
|
insertImageModal.modal('hide');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
insertButton.click(insertImage);
|
||||||
|
|
||||||
|
insertImageModal.on('shown', function() {
|
||||||
|
urlInput.focus();
|
||||||
|
});
|
||||||
|
|
||||||
|
insertImageModal.on('hide', function() {
|
||||||
|
self.editor.currentView.element.focus();
|
||||||
|
});
|
||||||
|
|
||||||
|
toolbar.find('a[data-wysihtml5-command=insertImage]').click(function() {
|
||||||
|
var activeButton = $(this).hasClass("wysihtml5-command-active");
|
||||||
|
|
||||||
|
if (!activeButton) {
|
||||||
|
self.editor.currentView.element.focus(false);
|
||||||
|
caretBookmark = self.editor.composer.selection.getBookmark();
|
||||||
|
insertImageModal.appendTo('body').modal('show');
|
||||||
|
insertImageModal.on('click.dismiss.modal', '[data-dismiss="modal"]', function(e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
initInsertLink: function(toolbar) {
|
||||||
|
var self = this;
|
||||||
|
var insertLinkModal = toolbar.find('.bootstrap-wysihtml5-insert-link-modal');
|
||||||
|
var urlInput = insertLinkModal.find('.bootstrap-wysihtml5-insert-link-url');
|
||||||
|
var targetInput = insertLinkModal.find('.bootstrap-wysihtml5-insert-link-target');
|
||||||
|
var insertButton = insertLinkModal.find('.btn-primary');
|
||||||
|
var initialValue = urlInput.val();
|
||||||
|
var caretBookmark;
|
||||||
|
|
||||||
|
var insertLink = function() {
|
||||||
|
var url = urlInput.val();
|
||||||
|
urlInput.val(initialValue);
|
||||||
|
self.editor.currentView.element.focus();
|
||||||
|
if (caretBookmark) {
|
||||||
|
self.editor.composer.selection.setBookmark(caretBookmark);
|
||||||
|
caretBookmark = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var newWindow = targetInput.prop("checked");
|
||||||
|
self.editor.composer.commands.exec("createLink", {
|
||||||
|
'href' : url,
|
||||||
|
'target' : (newWindow ? '_blank' : '_self'),
|
||||||
|
'rel' : (newWindow ? 'nofollow' : '')
|
||||||
|
});
|
||||||
|
};
|
||||||
|
var pressedEnter = false;
|
||||||
|
|
||||||
|
urlInput.keypress(function(e) {
|
||||||
|
if(e.which == 13) {
|
||||||
|
insertLink();
|
||||||
|
insertLinkModal.modal('hide');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
insertButton.click(insertLink);
|
||||||
|
|
||||||
|
insertLinkModal.on('shown', function() {
|
||||||
|
urlInput.focus();
|
||||||
|
});
|
||||||
|
|
||||||
|
insertLinkModal.on('hide', function() {
|
||||||
|
self.editor.currentView.element.focus();
|
||||||
|
});
|
||||||
|
|
||||||
|
toolbar.find('a[data-wysihtml5-command=createLink]').click(function() {
|
||||||
|
var activeButton = $(this).hasClass("wysihtml5-command-active");
|
||||||
|
|
||||||
|
if (!activeButton) {
|
||||||
|
self.editor.currentView.element.focus(false);
|
||||||
|
caretBookmark = self.editor.composer.selection.getBookmark();
|
||||||
|
insertLinkModal.appendTo('body').modal('show');
|
||||||
|
insertLinkModal.on('click.dismiss.modal', '[data-dismiss="modal"]', function(e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// these define our public api
|
||||||
|
var methods = {
|
||||||
|
resetDefaults: function() {
|
||||||
|
$.fn.wysihtml5.defaultOptions = $.extend(true, {}, $.fn.wysihtml5.defaultOptionsCache);
|
||||||
|
},
|
||||||
|
bypassDefaults: function(options) {
|
||||||
|
return this.each(function () {
|
||||||
|
var $this = $(this);
|
||||||
|
$this.data('wysihtml5', new Wysihtml5($this, options));
|
||||||
|
});
|
||||||
|
},
|
||||||
|
shallowExtend: function (options) {
|
||||||
|
var settings = $.extend({}, $.fn.wysihtml5.defaultOptions, options || {}, $(this).data());
|
||||||
|
var that = this;
|
||||||
|
return methods.bypassDefaults.apply(that, [settings]);
|
||||||
|
},
|
||||||
|
deepExtend: function(options) {
|
||||||
|
var settings = $.extend(true, {}, $.fn.wysihtml5.defaultOptions, options || {});
|
||||||
|
var that = this;
|
||||||
|
return methods.bypassDefaults.apply(that, [settings]);
|
||||||
|
},
|
||||||
|
init: function(options) {
|
||||||
|
var that = this;
|
||||||
|
return methods.shallowExtend.apply(that, [options]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$.fn.wysihtml5 = function ( method ) {
|
||||||
|
if ( methods[method] ) {
|
||||||
|
return methods[method].apply( this, Array.prototype.slice.call( arguments, 1 ));
|
||||||
|
} else if ( typeof method === 'object' || ! method ) {
|
||||||
|
return methods.init.apply( this, arguments );
|
||||||
|
} else {
|
||||||
|
$.error( 'Method ' + method + ' does not exist on jQuery.wysihtml5' );
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$.fn.wysihtml5.Constructor = Wysihtml5;
|
||||||
|
|
||||||
|
var defaultOptions = $.fn.wysihtml5.defaultOptions = {
|
||||||
|
"font-styles": true,
|
||||||
|
"color": false,
|
||||||
|
"emphasis": true,
|
||||||
|
"lists": true,
|
||||||
|
"html": false,
|
||||||
|
"link": true,
|
||||||
|
"image": true,
|
||||||
|
"size": 'sm',
|
||||||
|
events: {},
|
||||||
|
parserRules: {
|
||||||
|
classes: {
|
||||||
|
// (path_to_project/lib/css/bootstrap3-wysiwyg5-color.css)
|
||||||
|
"wysiwyg-color-silver" : 1,
|
||||||
|
"wysiwyg-color-gray" : 1,
|
||||||
|
"wysiwyg-color-white" : 1,
|
||||||
|
"wysiwyg-color-maroon" : 1,
|
||||||
|
"wysiwyg-color-red" : 1,
|
||||||
|
"wysiwyg-color-purple" : 1,
|
||||||
|
"wysiwyg-color-fuchsia" : 1,
|
||||||
|
"wysiwyg-color-green" : 1,
|
||||||
|
"wysiwyg-color-lime" : 1,
|
||||||
|
"wysiwyg-color-olive" : 1,
|
||||||
|
"wysiwyg-color-yellow" : 1,
|
||||||
|
"wysiwyg-color-navy" : 1,
|
||||||
|
"wysiwyg-color-blue" : 1,
|
||||||
|
"wysiwyg-color-teal" : 1,
|
||||||
|
"wysiwyg-color-aqua" : 1,
|
||||||
|
"wysiwyg-color-orange" : 1
|
||||||
|
},
|
||||||
|
tags: {
|
||||||
|
"b": {},
|
||||||
|
"i": {},
|
||||||
|
"br": {},
|
||||||
|
"ol": {},
|
||||||
|
"ul": {},
|
||||||
|
"li": {},
|
||||||
|
"h1": {},
|
||||||
|
"h2": {},
|
||||||
|
"h3": {},
|
||||||
|
"h4": {},
|
||||||
|
"h5": {},
|
||||||
|
"h6": {},
|
||||||
|
"blockquote": {},
|
||||||
|
"u": 1,
|
||||||
|
"img": {
|
||||||
|
"check_attributes": {
|
||||||
|
"width": "numbers",
|
||||||
|
"alt": "alt",
|
||||||
|
"src": "url",
|
||||||
|
"height": "numbers"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"a": {
|
||||||
|
check_attributes: {
|
||||||
|
'href': "url", // important to avoid XSS
|
||||||
|
'target': 'alt',
|
||||||
|
'rel': 'alt'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"span": 1,
|
||||||
|
"div": 1,
|
||||||
|
// to allow save and edit files with code tag hacks
|
||||||
|
"code": 1,
|
||||||
|
"pre": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
stylesheets: ["./css/bootstrap3-wysiwyg5-color.css"], // (path_to_project/lib/css/bootstrap3-wysiwyg5-color.css)
|
||||||
|
locale: "en"
|
||||||
|
};
|
||||||
|
|
||||||
|
if (typeof $.fn.wysihtml5.defaultOptionsCache === 'undefined') {
|
||||||
|
$.fn.wysihtml5.defaultOptionsCache = $.extend(true, {}, $.fn.wysihtml5.defaultOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
var locale = $.fn.wysihtml5.locale = {
|
||||||
|
en: {
|
||||||
|
font_styles: {
|
||||||
|
normal: "Normal text",
|
||||||
|
h1: "Heading 1",
|
||||||
|
h2: "Heading 2",
|
||||||
|
h3: "Heading 3",
|
||||||
|
h4: "Heading 4",
|
||||||
|
h5: "Heading 5",
|
||||||
|
h6: "Heading 6"
|
||||||
|
},
|
||||||
|
emphasis: {
|
||||||
|
bold: "Bold",
|
||||||
|
italic: "Italic",
|
||||||
|
underline: "Underline"
|
||||||
|
},
|
||||||
|
lists: {
|
||||||
|
unordered: "Unordered list",
|
||||||
|
ordered: "Ordered list",
|
||||||
|
outdent: "Outdent",
|
||||||
|
indent: "Indent"
|
||||||
|
},
|
||||||
|
link: {
|
||||||
|
insert: "Insert link",
|
||||||
|
cancel: "Cancel",
|
||||||
|
target: "Open link in new window"
|
||||||
|
},
|
||||||
|
image: {
|
||||||
|
insert: "Insert image",
|
||||||
|
cancel: "Cancel"
|
||||||
|
},
|
||||||
|
html: {
|
||||||
|
edit: "Edit HTML"
|
||||||
|
},
|
||||||
|
colours: {
|
||||||
|
black: "Black",
|
||||||
|
silver: "Silver",
|
||||||
|
gray: "Grey",
|
||||||
|
maroon: "Maroon",
|
||||||
|
red: "Red",
|
||||||
|
purple: "Purple",
|
||||||
|
green: "Green",
|
||||||
|
olive: "Olive",
|
||||||
|
navy: "Navy",
|
||||||
|
blue: "Blue",
|
||||||
|
orange: "Orange"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
}(window.jQuery, window.wysihtml5);
|
315
js/ekko-lightbox-modified.js
Normal file
@ -0,0 +1,315 @@
|
|||||||
|
// Generated by CoffeeScript 1.6.3
|
||||||
|
/*
|
||||||
|
Lightbox for Bootstrap 3 by @ashleydw
|
||||||
|
https://github.com/ashleydw/lightbox
|
||||||
|
|
||||||
|
License: https://github.com/ashleydw/lightbox/blob/master/LICENSE
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
"use strict";
|
||||||
|
var EkkoLightbox;
|
||||||
|
|
||||||
|
EkkoLightbox = function (element, options) {
|
||||||
|
var content, footer, header, video_id,
|
||||||
|
_this = this;
|
||||||
|
this.options = $.extend({
|
||||||
|
gallery_parent_selector: '*:not(.row)',
|
||||||
|
title: null,
|
||||||
|
footer: null,
|
||||||
|
remote: null,
|
||||||
|
left_arrow_class: '.glyphicon .glyphicon-chevron-left',
|
||||||
|
right_arrow_class: '.glyphicon .glyphicon-chevron-right',
|
||||||
|
directional_arrows: true,
|
||||||
|
type: null,
|
||||||
|
onShow: function () {
|
||||||
|
},
|
||||||
|
onShown: function () {
|
||||||
|
},
|
||||||
|
onHide: function () {
|
||||||
|
},
|
||||||
|
onHidden: function () {
|
||||||
|
},
|
||||||
|
id: false
|
||||||
|
}, options || {});
|
||||||
|
this.$element = $(element);
|
||||||
|
content = '';
|
||||||
|
this.modal_id = this.options.modal_id ? this.options.modal_id : 'ekkoLightbox-' + Math.floor((Math.random() * 1000) + 1);
|
||||||
|
header = '<div class="modal-header"' + (this.options.title ? '' : ' style="display:none"') + '><button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button><h4 class="modal-title">' + this.options.title + '</h4></div>';
|
||||||
|
footer = '<div class="modal-footer"' + (this.options.footer ? '' : ' style="display:none"') + '>' + this.options.footer + '</div>';
|
||||||
|
//footer = '<div class="modal-footer"><button type="button" class="btn btn-primary" data-dismiss="modal">Close</button></div>';
|
||||||
|
$(document.body).append('<div id="' + this.modal_id + '" class="ekko-lightbox modal" tabindex="-1"><div class="modal-dialog" style="margin-top: 0 !important"><div class="modal-content">' + header + '<div class="modal-body"><div class="ekko-lightbox-container"><div></div></div></div>' + footer + '</div></div></div>');
|
||||||
|
this.modal = $('#' + this.modal_id);
|
||||||
|
this.modal_body = this.modal.find('.modal-body').first();
|
||||||
|
this.lightbox_container = this.modal_body.find('.ekko-lightbox-container').first();
|
||||||
|
this.lightbox_body = this.lightbox_container.find('> div:first-child').first();
|
||||||
|
this.modal_arrows = null;
|
||||||
|
this.padding = {
|
||||||
|
left: parseFloat(this.modal_body.css('padding-left'), 10),
|
||||||
|
right: parseFloat(this.modal_body.css('padding-right'), 10),
|
||||||
|
bottom: parseFloat(this.modal_body.css('padding-bottom'), 10),
|
||||||
|
top: parseFloat(this.modal_body.css('padding-top'), 10)
|
||||||
|
};
|
||||||
|
if (!this.options.remote) {
|
||||||
|
this.error('No remote target given');
|
||||||
|
} else {
|
||||||
|
this.gallery = this.$element.data('gallery');
|
||||||
|
if (this.gallery) {
|
||||||
|
if (this.options.gallery_parent_selector === 'document.body' || this.options.gallery_parent_selector === '') {
|
||||||
|
this.gallery_items = $(document.body).find('*[data-toggle="lightbox"][data-gallery="' + this.gallery + '"]');
|
||||||
|
} else {
|
||||||
|
this.gallery_items = this.$element.parents(this.options.gallery_parent_selector).first().find('*[data-toggle="lightbox"][data-gallery="' + this.gallery + '"]');
|
||||||
|
}
|
||||||
|
this.gallery_index = this.gallery_items.index(this.$element);
|
||||||
|
$(document).on('keydown.ekkoLightbox', this.navigate.bind(this));
|
||||||
|
if (this.options.directional_arrows) {
|
||||||
|
this.lightbox_container.prepend('<div class="ekko-lightbox-nav-overlay"><a href="#" class="' + this.strip_stops(this.options.left_arrow_class) + '"></a><a href="#" class="' + this.strip_stops(this.options.right_arrow_class) + '"></a></div>');
|
||||||
|
this.modal_arrows = this.lightbox_container.find('div.ekko-lightbox-nav-overlay').first();
|
||||||
|
this.lightbox_container.find('a' + this.strip_spaces(this.options.left_arrow_class)).on('click', function (event) {
|
||||||
|
event.preventDefault();
|
||||||
|
return _this.navigate_left();
|
||||||
|
});
|
||||||
|
this.lightbox_container.find('a' + this.strip_spaces(this.options.right_arrow_class)).on('click', function (event) {
|
||||||
|
event.preventDefault();
|
||||||
|
return _this.navigate_right();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this.options.type) {
|
||||||
|
if (this.options.type === 'image') {
|
||||||
|
this.preloadImage(this.options.remote, true);
|
||||||
|
} else if (this.options.type === 'youtube' && (video_id = this.getYoutubeId(this.options.remote))) {
|
||||||
|
this.showYoutubeVideo(video_id);
|
||||||
|
} else if (this.options.type === 'vimeo') {
|
||||||
|
this.showVimeoVideo(this.options.remote);
|
||||||
|
} else {
|
||||||
|
this.error("Could not detect remote target type. Force the type using data-type=\"image|youtube|vimeo\"");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.detectRemoteType(this.options.remote);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.modal.on('show.bs.modal', this.options.onShow.bind(this)).on('shown.bs.modal',function () {
|
||||||
|
if (_this.modal_arrows) {
|
||||||
|
_this.resize(_this.lightbox_body.width());
|
||||||
|
}
|
||||||
|
return _this.options.onShown.call(_this);
|
||||||
|
}).on('hide.bs.modal', this.options.onHide.bind(this)).on('hidden.bs.modal',function () {
|
||||||
|
if (_this.gallery) {
|
||||||
|
$(document).off('keydown.ekkoLightbox');
|
||||||
|
}
|
||||||
|
_this.modal.remove();
|
||||||
|
return _this.options.onHidden.call(_this);
|
||||||
|
}).modal('show', options);
|
||||||
|
return this.modal;
|
||||||
|
};
|
||||||
|
|
||||||
|
EkkoLightbox.prototype = {
|
||||||
|
strip_stops: function (str) {
|
||||||
|
return str.replace(/\./g, '');
|
||||||
|
},
|
||||||
|
strip_spaces: function (str) {
|
||||||
|
return str.replace(/\s/g, '');
|
||||||
|
},
|
||||||
|
isImage: function (str) {
|
||||||
|
return str.match(/(^data:image\/.*,)|(\.(jp(e|g|eg)|gif|png|bmp|webp|svg)((\?|#).*)?$)/i);
|
||||||
|
},
|
||||||
|
isSwf: function (str) {
|
||||||
|
return str.match(/\.(swf)((\?|#).*)?$/i);
|
||||||
|
},
|
||||||
|
getYoutubeId: function (str) {
|
||||||
|
var match;
|
||||||
|
match = str.match(/^.*(youtu.be\/|v\/|u\/\w\/|embed\/|watch\?v=|\&v=)([^#\&\?]*).*/);
|
||||||
|
if (match && match[2].length === 11) {
|
||||||
|
return match[2];
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getVimeoId: function (str) {
|
||||||
|
if (str.indexOf('vimeo') > 0) {
|
||||||
|
return str;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
navigate: function (event) {
|
||||||
|
event = event || window.event;
|
||||||
|
if (event.keyCode === 39 || event.keyCode === 37) {
|
||||||
|
if (event.keyCode === 39) {
|
||||||
|
return this.navigate_right();
|
||||||
|
} else if (event.keyCode === 37) {
|
||||||
|
return this.navigate_left();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
navigate_left: function () {
|
||||||
|
var src;
|
||||||
|
if (this.gallery_index === 0) {
|
||||||
|
this.gallery_index = this.gallery_items.length - 1;
|
||||||
|
} else {
|
||||||
|
this.gallery_index--;
|
||||||
|
}
|
||||||
|
this.$element = $(this.gallery_items.get(this.gallery_index));
|
||||||
|
this.updateTitleAndFooter();
|
||||||
|
src = this.$element.attr('data-remote') || this.$element.attr('href');
|
||||||
|
return this.detectRemoteType(src);
|
||||||
|
},
|
||||||
|
navigate_right: function () {
|
||||||
|
var next, src;
|
||||||
|
if (this.gallery_index === this.gallery_items.length - 1) {
|
||||||
|
this.gallery_index = 0;
|
||||||
|
} else {
|
||||||
|
this.gallery_index++;
|
||||||
|
}
|
||||||
|
this.$element = $(this.gallery_items.get(this.gallery_index));
|
||||||
|
src = this.$element.attr('data-remote') || this.$element.attr('href');
|
||||||
|
this.updateTitleAndFooter();
|
||||||
|
this.detectRemoteType(src);
|
||||||
|
if (this.gallery_index + 1 < this.gallery_items.length) {
|
||||||
|
next = $(this.gallery_items.get(this.gallery_index + 1), false);
|
||||||
|
src = next.attr('data-remote') || next.attr('href');
|
||||||
|
if (this.isImage(src)) {
|
||||||
|
return this.preloadImage(src, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
detectRemoteType: function (src) {
|
||||||
|
var video_id;
|
||||||
|
if (this.isImage(src)) {
|
||||||
|
return this.preloadImage(src, true);
|
||||||
|
} else if (video_id = this.getYoutubeId(src)) {
|
||||||
|
return this.showYoutubeVideo(video_id);
|
||||||
|
} else if (video_id = this.getVimeoId(src)) {
|
||||||
|
return this.showVimeoVideo(video_id);
|
||||||
|
} else {
|
||||||
|
return this.error("Could not detect remote target type. Force the type using data-type=\"image|youtube|vimeo\"");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
updateTitleAndFooter: function () {
|
||||||
|
var caption, footer, header, title;
|
||||||
|
header = this.modal.find('.modal-dialog .modal-content .modal-header');
|
||||||
|
footer = this.modal.find('.modal-dialog .modal-content .modal-footer');
|
||||||
|
title = this.$element.data('title') || "";
|
||||||
|
caption = this.$element.data('footer') || "";
|
||||||
|
if (title) {
|
||||||
|
header.css('display', '').find('.modal-title').html(title);
|
||||||
|
} else {
|
||||||
|
header.css('display', 'none');
|
||||||
|
}
|
||||||
|
if (caption) {
|
||||||
|
footer.css('display', '').html(caption);
|
||||||
|
} else {
|
||||||
|
footer.css('display', 'none');
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
showLoading: function () {
|
||||||
|
this.lightbox_body.html('<div class="modal-loading"><div class="loader"></div></div>');
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
showYoutubeVideo: function (id) {
|
||||||
|
this.resize(560);
|
||||||
|
this.lightbox_body.html('<iframe width="560" height="315" src="//www.youtube.com/embed/' + id + '?badge=0&autoplay=1" frameborder="0" allowfullscreen></iframe>');
|
||||||
|
if (this.modal_arrows) {
|
||||||
|
return this.modal_arrows.css('display', 'none');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
showVimeoVideo: function (id) {
|
||||||
|
this.resize(500);
|
||||||
|
this.lightbox_body.html('<iframe width="500" height="281" src="' + id + '?autoplay=1" frameborder="0" allowfullscreen></iframe>');
|
||||||
|
if (this.modal_arrows) {
|
||||||
|
return this.modal_arrows.css('display', 'none');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
error: function (message) {
|
||||||
|
this.lightbox_body.html(message);
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
preloadImage: function (src, onLoadShowImage) {
|
||||||
|
|
||||||
|
var img,
|
||||||
|
_this = this;
|
||||||
|
|
||||||
|
_this.showLoading();
|
||||||
|
|
||||||
|
img = new Image();
|
||||||
|
if ((onLoadShowImage == null) || onLoadShowImage === true) {
|
||||||
|
img.onload = function () {
|
||||||
|
var image, width;
|
||||||
|
width = _this.checkImageDimensions(img.width);
|
||||||
|
image = $('<img />');
|
||||||
|
image.attr('src', img.src);
|
||||||
|
image.css('max-width', '100%');
|
||||||
|
_this.lightbox_body.html(image);
|
||||||
|
if (_this.modal_arrows) {
|
||||||
|
_this.modal_arrows.css('display', 'block');
|
||||||
|
}
|
||||||
|
return _this.resize(width);
|
||||||
|
};
|
||||||
|
img.onerror = function () {
|
||||||
|
return _this.error('Failed to load image: ' + src);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
img.src = src;
|
||||||
|
return img;
|
||||||
|
},
|
||||||
|
resize: function (width) {
|
||||||
|
var width_inc_padding;
|
||||||
|
width_inc_padding = width + this.padding.left + this.padding.right;
|
||||||
|
//this.modal.find('.modal-content').css('width', width_inc_padding);
|
||||||
|
//this.modal.find('.modal-dialog').css('width', width_inc_padding + 20);
|
||||||
|
this.modal.find('.modal-content').animate({width: width_inc_padding}, 200);
|
||||||
|
this.modal.find('.modal-dialog').animate({width: width_inc_padding + 20}, 200);
|
||||||
|
this.lightbox_container.find('a').css('padding-top', function () {
|
||||||
|
return $(this).parent().height() / 2;
|
||||||
|
});
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
checkImageDimensions: function (max_width) {
|
||||||
|
var w, width;
|
||||||
|
w = $(window);
|
||||||
|
width = max_width;
|
||||||
|
if ((max_width + (this.padding.left + this.padding.right + 20)) > w.width()) {
|
||||||
|
width = w.width() - (this.padding.left + this.padding.right + 20);
|
||||||
|
}
|
||||||
|
return width - 20;
|
||||||
|
},
|
||||||
|
close: function () {
|
||||||
|
return this.modal.modal('hide');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$.fn.ekkoLightbox = function (options) {
|
||||||
|
return this.each(function () {
|
||||||
|
var $this;
|
||||||
|
$this = $(this);
|
||||||
|
options = $.extend({
|
||||||
|
remote: $this.attr('data-remote') || $this.attr('href'),
|
||||||
|
gallery_parent_selector: $this.attr('data-parent'),
|
||||||
|
type: $this.attr('data-type')
|
||||||
|
}, options, $this.data());
|
||||||
|
new EkkoLightbox(this, options);
|
||||||
|
return this;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
$(document).delegate('*[data-toggle="lightbox"]', 'click', function (event) {
|
||||||
|
var $this;
|
||||||
|
event.preventDefault();
|
||||||
|
$this = $(this);
|
||||||
|
return $this.ekkoLightbox({
|
||||||
|
remote: $this.attr('data-remote') || $this.attr('href'),
|
||||||
|
gallery_parent_selector: $this.attr('data-parent'),
|
||||||
|
onShown: function () {
|
||||||
|
if (window.console) {
|
||||||
|
return console.log('Checking our the events huh?');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}).one('hide', function () {
|
||||||
|
return $this.is(':visible') && $this.focus();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
}).call(this);
|
301
js/html5shiv.js
vendored
Normal file
@ -0,0 +1,301 @@
|
|||||||
|
/**
|
||||||
|
* @preserve HTML5 Shiv prev3.7.1 | @afarkas @jdalton @jon_neal @rem | MIT/GPL2 Licensed
|
||||||
|
*/
|
||||||
|
;(function(window, document) {
|
||||||
|
/*jshint evil:true */
|
||||||
|
/** version */
|
||||||
|
var version = '3.7.0';
|
||||||
|
|
||||||
|
/** Preset options */
|
||||||
|
var options = window.html5 || {};
|
||||||
|
|
||||||
|
/** Used to skip problem elements */
|
||||||
|
var reSkip = /^<|^(?:button|map|select|textarea|object|iframe|option|optgroup)$/i;
|
||||||
|
|
||||||
|
/** Not all elements can be cloned in IE **/
|
||||||
|
var saveClones = /^(?:a|b|code|div|fieldset|h1|h2|h3|h4|h5|h6|i|label|li|ol|p|q|span|strong|style|table|tbody|td|th|tr|ul)$/i;
|
||||||
|
|
||||||
|
/** Detect whether the browser supports default html5 styles */
|
||||||
|
var supportsHtml5Styles;
|
||||||
|
|
||||||
|
/** Name of the expando, to work with multiple documents or to re-shiv one document */
|
||||||
|
var expando = '_html5shiv';
|
||||||
|
|
||||||
|
/** The id for the the documents expando */
|
||||||
|
var expanID = 0;
|
||||||
|
|
||||||
|
/** Cached data for each document */
|
||||||
|
var expandoData = {};
|
||||||
|
|
||||||
|
/** Detect whether the browser supports unknown elements */
|
||||||
|
var supportsUnknownElements;
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
try {
|
||||||
|
var a = document.createElement('a');
|
||||||
|
a.innerHTML = '<xyz></xyz>';
|
||||||
|
//if the hidden property is implemented we can assume, that the browser supports basic HTML5 Styles
|
||||||
|
supportsHtml5Styles = ('hidden' in a);
|
||||||
|
|
||||||
|
supportsUnknownElements = a.childNodes.length == 1 || (function() {
|
||||||
|
// assign a false positive if unable to shiv
|
||||||
|
(document.createElement)('a');
|
||||||
|
var frag = document.createDocumentFragment();
|
||||||
|
return (
|
||||||
|
typeof frag.cloneNode == 'undefined' ||
|
||||||
|
typeof frag.createDocumentFragment == 'undefined' ||
|
||||||
|
typeof frag.createElement == 'undefined'
|
||||||
|
);
|
||||||
|
}());
|
||||||
|
} catch(e) {
|
||||||
|
// assign a false positive if detection fails => unable to shiv
|
||||||
|
supportsHtml5Styles = true;
|
||||||
|
supportsUnknownElements = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}());
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a style sheet with the given CSS text and adds it to the document.
|
||||||
|
* @private
|
||||||
|
* @param {Document} ownerDocument The document.
|
||||||
|
* @param {String} cssText The CSS text.
|
||||||
|
* @returns {StyleSheet} The style element.
|
||||||
|
*/
|
||||||
|
function addStyleSheet(ownerDocument, cssText) {
|
||||||
|
var p = ownerDocument.createElement('p'),
|
||||||
|
parent = ownerDocument.getElementsByTagName('head')[0] || ownerDocument.documentElement;
|
||||||
|
|
||||||
|
p.innerHTML = 'x<style>' + cssText + '</style>';
|
||||||
|
return parent.insertBefore(p.lastChild, parent.firstChild);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the value of `html5.elements` as an array.
|
||||||
|
* @private
|
||||||
|
* @returns {Array} An array of shived element node names.
|
||||||
|
*/
|
||||||
|
function getElements() {
|
||||||
|
var elements = html5.elements;
|
||||||
|
return typeof elements == 'string' ? elements.split(' ') : elements;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the data associated to the given document
|
||||||
|
* @private
|
||||||
|
* @param {Document} ownerDocument The document.
|
||||||
|
* @returns {Object} An object of data.
|
||||||
|
*/
|
||||||
|
function getExpandoData(ownerDocument) {
|
||||||
|
var data = expandoData[ownerDocument[expando]];
|
||||||
|
if (!data) {
|
||||||
|
data = {};
|
||||||
|
expanID++;
|
||||||
|
ownerDocument[expando] = expanID;
|
||||||
|
expandoData[expanID] = data;
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns a shived element for the given nodeName and document
|
||||||
|
* @memberOf html5
|
||||||
|
* @param {String} nodeName name of the element
|
||||||
|
* @param {Document} ownerDocument The context document.
|
||||||
|
* @returns {Object} The shived element.
|
||||||
|
*/
|
||||||
|
function createElement(nodeName, ownerDocument, data){
|
||||||
|
if (!ownerDocument) {
|
||||||
|
ownerDocument = document;
|
||||||
|
}
|
||||||
|
if(supportsUnknownElements){
|
||||||
|
return ownerDocument.createElement(nodeName);
|
||||||
|
}
|
||||||
|
if (!data) {
|
||||||
|
data = getExpandoData(ownerDocument);
|
||||||
|
}
|
||||||
|
var node;
|
||||||
|
|
||||||
|
if (data.cache[nodeName]) {
|
||||||
|
node = data.cache[nodeName].cloneNode();
|
||||||
|
} else if (saveClones.test(nodeName)) {
|
||||||
|
node = (data.cache[nodeName] = data.createElem(nodeName)).cloneNode();
|
||||||
|
} else {
|
||||||
|
node = data.createElem(nodeName);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Avoid adding some elements to fragments in IE < 9 because
|
||||||
|
// * Attributes like `name` or `type` cannot be set/changed once an element
|
||||||
|
// is inserted into a document/fragment
|
||||||
|
// * Link elements with `src` attributes that are inaccessible, as with
|
||||||
|
// a 403 response, will cause the tab/window to crash
|
||||||
|
// * Script elements appended to fragments will execute when their `src`
|
||||||
|
// or `text` property is set
|
||||||
|
return node.canHaveChildren && !reSkip.test(nodeName) && !node.tagUrn ? data.frag.appendChild(node) : node;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns a shived DocumentFragment for the given document
|
||||||
|
* @memberOf html5
|
||||||
|
* @param {Document} ownerDocument The context document.
|
||||||
|
* @returns {Object} The shived DocumentFragment.
|
||||||
|
*/
|
||||||
|
function createDocumentFragment(ownerDocument, data){
|
||||||
|
if (!ownerDocument) {
|
||||||
|
ownerDocument = document;
|
||||||
|
}
|
||||||
|
if(supportsUnknownElements){
|
||||||
|
return ownerDocument.createDocumentFragment();
|
||||||
|
}
|
||||||
|
data = data || getExpandoData(ownerDocument);
|
||||||
|
var clone = data.frag.cloneNode(),
|
||||||
|
i = 0,
|
||||||
|
elems = getElements(),
|
||||||
|
l = elems.length;
|
||||||
|
for(;i<l;i++){
|
||||||
|
clone.createElement(elems[i]);
|
||||||
|
}
|
||||||
|
return clone;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Shivs the `createElement` and `createDocumentFragment` methods of the document.
|
||||||
|
* @private
|
||||||
|
* @param {Document|DocumentFragment} ownerDocument The document.
|
||||||
|
* @param {Object} data of the document.
|
||||||
|
*/
|
||||||
|
function shivMethods(ownerDocument, data) {
|
||||||
|
if (!data.cache) {
|
||||||
|
data.cache = {};
|
||||||
|
data.createElem = ownerDocument.createElement;
|
||||||
|
data.createFrag = ownerDocument.createDocumentFragment;
|
||||||
|
data.frag = data.createFrag();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ownerDocument.createElement = function(nodeName) {
|
||||||
|
//abort shiv
|
||||||
|
if (!html5.shivMethods) {
|
||||||
|
return data.createElem(nodeName);
|
||||||
|
}
|
||||||
|
return createElement(nodeName, ownerDocument, data);
|
||||||
|
};
|
||||||
|
|
||||||
|
ownerDocument.createDocumentFragment = Function('h,f', 'return function(){' +
|
||||||
|
'var n=f.cloneNode(),c=n.createElement;' +
|
||||||
|
'h.shivMethods&&(' +
|
||||||
|
// unroll the `createElement` calls
|
||||||
|
getElements().join().replace(/[\w\-]+/g, function(nodeName) {
|
||||||
|
data.createElem(nodeName);
|
||||||
|
data.frag.createElement(nodeName);
|
||||||
|
return 'c("' + nodeName + '")';
|
||||||
|
}) +
|
||||||
|
');return n}'
|
||||||
|
)(html5, data.frag);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Shivs the given document.
|
||||||
|
* @memberOf html5
|
||||||
|
* @param {Document} ownerDocument The document to shiv.
|
||||||
|
* @returns {Document} The shived document.
|
||||||
|
*/
|
||||||
|
function shivDocument(ownerDocument) {
|
||||||
|
if (!ownerDocument) {
|
||||||
|
ownerDocument = document;
|
||||||
|
}
|
||||||
|
var data = getExpandoData(ownerDocument);
|
||||||
|
|
||||||
|
if (html5.shivCSS && !supportsHtml5Styles && !data.hasCSS) {
|
||||||
|
data.hasCSS = !!addStyleSheet(ownerDocument,
|
||||||
|
// corrects block display not defined in IE6/7/8/9
|
||||||
|
'article,aside,dialog,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}' +
|
||||||
|
// adds styling not present in IE6/7/8/9
|
||||||
|
'mark{background:#FF0;color:#000}' +
|
||||||
|
// hides non-rendered elements
|
||||||
|
'template{display:none}'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (!supportsUnknownElements) {
|
||||||
|
shivMethods(ownerDocument, data);
|
||||||
|
}
|
||||||
|
return ownerDocument;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The `html5` object is exposed so that more elements can be shived and
|
||||||
|
* existing shiving can be detected on iframes.
|
||||||
|
* @type Object
|
||||||
|
* @example
|
||||||
|
*
|
||||||
|
* // options can be changed before the script is included
|
||||||
|
* html5 = { 'elements': 'mark section', 'shivCSS': false, 'shivMethods': false };
|
||||||
|
*/
|
||||||
|
var html5 = {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An array or space separated string of node names of the elements to shiv.
|
||||||
|
* @memberOf html5
|
||||||
|
* @type Array|String
|
||||||
|
*/
|
||||||
|
'elements': options.elements || 'abbr article aside audio bdi canvas data datalist details dialog figcaption figure footer header hgroup main mark meter nav output progress section summary template time video',
|
||||||
|
|
||||||
|
/**
|
||||||
|
* current version of html5shiv
|
||||||
|
*/
|
||||||
|
'version': version,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A flag to indicate that the HTML5 style sheet should be inserted.
|
||||||
|
* @memberOf html5
|
||||||
|
* @type Boolean
|
||||||
|
*/
|
||||||
|
'shivCSS': (options.shivCSS !== false),
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is equal to true if a browser supports creating unknown/HTML5 elements
|
||||||
|
* @memberOf html5
|
||||||
|
* @type boolean
|
||||||
|
*/
|
||||||
|
'supportsUnknownElements': supportsUnknownElements,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A flag to indicate that the document's `createElement` and `createDocumentFragment`
|
||||||
|
* methods should be overwritten.
|
||||||
|
* @memberOf html5
|
||||||
|
* @type Boolean
|
||||||
|
*/
|
||||||
|
'shivMethods': (options.shivMethods !== false),
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A string to describe the type of `html5` object ("default" or "default print").
|
||||||
|
* @memberOf html5
|
||||||
|
* @type String
|
||||||
|
*/
|
||||||
|
'type': 'default',
|
||||||
|
|
||||||
|
// shivs the document according to the specified `html5` object options
|
||||||
|
'shivDocument': shivDocument,
|
||||||
|
|
||||||
|
//creates a shived element
|
||||||
|
createElement: createElement,
|
||||||
|
|
||||||
|
//creates a shived documentFragment
|
||||||
|
createDocumentFragment: createDocumentFragment
|
||||||
|
};
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
// expose html5
|
||||||
|
window.html5 = html5;
|
||||||
|
|
||||||
|
// shiv the document
|
||||||
|
shivDocument(document);
|
||||||
|
|
||||||
|
}(this, document));
|
7
js/jquery.autosize.min.js
vendored
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
/*!
|
||||||
|
Autosize v1.17.8 - 2013-09-07
|
||||||
|
Automatically adjust textarea height based on user input.
|
||||||
|
(c) 2013 Jack Moore - http://www.jacklmoore.com/autosize
|
||||||
|
license: http://www.opensource.org/licenses/mit-license.php
|
||||||
|
*/
|
||||||
|
(function(e){"function"==typeof define&&define.amd?define(["jquery"],e):e(window.jQuery||window.$)})(function(e){var t,o={className:"autosizejs",append:"",callback:!1,resizeDelay:10},i='<textarea tabindex="-1" style="position:absolute; top:-999px; left:0; right:auto; bottom:auto; border:0; padding: 0; -moz-box-sizing:content-box; -webkit-box-sizing:content-box; box-sizing:content-box; word-wrap:break-word; height:0 !important; min-height:0 !important; overflow:hidden; transition:none; -webkit-transition:none; -moz-transition:none;"/>',n=["fontFamily","fontSize","fontWeight","fontStyle","letterSpacing","textTransform","wordSpacing","textIndent"],s=e(i).data("autosize",!0)[0];s.style.lineHeight="99px","99px"===e(s).css("lineHeight")&&n.push("lineHeight"),s.style.lineHeight="",e.fn.autosize=function(i){return this.length?(i=e.extend({},o,i||{}),s.parentNode!==document.body&&e(document.body).append(s),this.each(function(){function o(){var t,o;"getComputedStyle"in window?(t=window.getComputedStyle(u),o=u.getBoundingClientRect().width,e.each(["paddingLeft","paddingRight","borderLeftWidth","borderRightWidth"],function(e,i){o-=parseInt(t[i],10)}),s.style.width=o+"px"):s.style.width=Math.max(p.width(),0)+"px"}function a(){var a={};if(t=u,s.className=i.className,d=parseInt(p.css("maxHeight"),10),e.each(n,function(e,t){a[t]=p.css(t)}),e(s).css(a),o(),window.chrome){var r=u.style.width;u.style.width="0px",u.offsetWidth,u.style.width=r}}function r(){var e,n;t!==u?a():o(),s.value=u.value+i.append,s.style.overflowY=u.style.overflowY,n=parseInt(u.style.height,10),s.scrollTop=0,s.scrollTop=9e4,e=s.scrollTop,d&&e>d?(u.style.overflowY="scroll",e=d):(u.style.overflowY="hidden",c>e&&(e=c)),e+=f,n!==e&&(u.style.height=e+"px",w&&i.callback.call(u,u))}function l(){clearTimeout(h),h=setTimeout(function(){var e=p.width();e!==g&&(g=e,r())},parseInt(i.resizeDelay,10))}var d,c,h,u=this,p=e(u),f=0,w=e.isFunction(i.callback),z={height:u.style.height,overflow:u.style.overflow,overflowY:u.style.overflowY,wordWrap:u.style.wordWrap,resize:u.style.resize},g=p.width();p.data("autosize")||(p.data("autosize",!0),("border-box"===p.css("box-sizing")||"border-box"===p.css("-moz-box-sizing")||"border-box"===p.css("-webkit-box-sizing"))&&(f=p.outerHeight()-p.height()),c=Math.max(parseInt(p.css("minHeight"),10)-f||0,p.height()),p.css({overflow:"hidden",overflowY:"hidden",wordWrap:"break-word",resize:"none"===p.css("resize")||"vertical"===p.css("resize")?"none":"horizontal"}),"onpropertychange"in u?"oninput"in u?p.on("input.autosize keyup.autosize",r):p.on("propertychange.autosize",function(){"value"===event.propertyName&&r()}):p.on("input.autosize",r),i.resizeDelay!==!1&&e(window).on("resize.autosize",l),p.on("autosize.resize",r),p.on("autosize.resizeIncludeStyle",function(){t=null,r()}),p.on("autosize.destroy",function(){t=null,clearTimeout(h),e(window).off("resize",l),p.off("autosize").off(".autosize").css(z).removeData("autosize")}),r())})):this}});
|
113
js/jquery.cookie.js
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
/*!
|
||||||
|
* jQuery Cookie Plugin v1.4.0
|
||||||
|
* https://github.com/carhartl/jquery-cookie
|
||||||
|
*
|
||||||
|
* Copyright 2013 Klaus Hartl
|
||||||
|
* Released under the MIT license
|
||||||
|
*/
|
||||||
|
(function (factory) {
|
||||||
|
if (typeof define === 'function' && define.amd) {
|
||||||
|
// AMD. Register as anonymous module.
|
||||||
|
define(['jquery'], factory);
|
||||||
|
} else {
|
||||||
|
// Browser globals.
|
||||||
|
factory(jQuery);
|
||||||
|
}
|
||||||
|
}(function ($) {
|
||||||
|
|
||||||
|
var pluses = /\+/g;
|
||||||
|
|
||||||
|
function encode(s) {
|
||||||
|
return config.raw ? s : encodeURIComponent(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
function decode(s) {
|
||||||
|
return config.raw ? s : decodeURIComponent(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
function stringifyCookieValue(value) {
|
||||||
|
return encode(config.json ? JSON.stringify(value) : String(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseCookieValue(s) {
|
||||||
|
if (s.indexOf('"') === 0) {
|
||||||
|
// This is a quoted cookie as according to RFC2068, unescape...
|
||||||
|
s = s.slice(1, -1).replace(/\\"/g, '"').replace(/\\\\/g, '\\');
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Replace server-side written pluses with spaces.
|
||||||
|
// If we can't decode the cookie, ignore it, it's unusable.
|
||||||
|
// If we can't parse the cookie, ignore it, it's unusable.
|
||||||
|
s = decodeURIComponent(s.replace(pluses, ' '));
|
||||||
|
return config.json ? JSON.parse(s) : s;
|
||||||
|
} catch(e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
function read(s, converter) {
|
||||||
|
var value = config.raw ? s : parseCookieValue(s);
|
||||||
|
return $.isFunction(converter) ? converter(value) : value;
|
||||||
|
}
|
||||||
|
|
||||||
|
var config = $.cookie = function (key, value, options) {
|
||||||
|
|
||||||
|
// Write
|
||||||
|
if (value !== undefined && !$.isFunction(value)) {
|
||||||
|
options = $.extend({}, config.defaults, options);
|
||||||
|
|
||||||
|
if (typeof options.expires === 'number') {
|
||||||
|
var days = options.expires, t = options.expires = new Date();
|
||||||
|
t.setDate(t.getDate() + days);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (document.cookie = [
|
||||||
|
encode(key), '=', stringifyCookieValue(value),
|
||||||
|
options.expires ? '; expires=' + options.expires.toUTCString() : '', // use expires attribute, max-age is not supported by IE
|
||||||
|
options.path ? '; path=' + options.path : '',
|
||||||
|
options.domain ? '; domain=' + options.domain : '',
|
||||||
|
options.secure ? '; secure' : ''
|
||||||
|
].join(''));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read
|
||||||
|
|
||||||
|
var result = key ? undefined : {};
|
||||||
|
|
||||||
|
// To prevent the for loop in the first place assign an empty array
|
||||||
|
// in case there are no cookies at all. Also prevents odd result when
|
||||||
|
// calling $.cookie().
|
||||||
|
var cookies = document.cookie ? document.cookie.split('; ') : [];
|
||||||
|
|
||||||
|
for (var i = 0, l = cookies.length; i < l; i++) {
|
||||||
|
var parts = cookies[i].split('=');
|
||||||
|
var name = decode(parts.shift());
|
||||||
|
var cookie = parts.join('=');
|
||||||
|
|
||||||
|
if (key && key === name) {
|
||||||
|
// If second argument (value) is a function it's a converter...
|
||||||
|
result = read(cookie, value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prevent storing a cookie that we couldn't decode.
|
||||||
|
if (!key && (cookie = read(cookie)) !== undefined) {
|
||||||
|
result[name] = cookie;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
config.defaults = {};
|
||||||
|
|
||||||
|
$.removeCookie = function (key, options) {
|
||||||
|
if ($.cookie(key) === undefined) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Must not alter options, thus extending a fresh object...
|
||||||
|
$.cookie(key, '', $.extend({}, options, { expires: -1 }));
|
||||||
|
return !$.cookie(key);
|
||||||
|
};
|
||||||
|
|
||||||
|
}));
|
17
js/jquery.highlight.min.js
vendored
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
highlight v4
|
||||||
|
|
||||||
|
Highlights arbitrary terms.
|
||||||
|
|
||||||
|
<http://johannburkard.de/blog/programming/javascript/highlight-javascript-text-higlighting-jquery-plugin.html>
|
||||||
|
|
||||||
|
MIT license.
|
||||||
|
|
||||||
|
Johann Burkard
|
||||||
|
<http://johannburkard.de>
|
||||||
|
<mailto:jb@eaio.com>
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
jQuery.fn.highlight=function(c){function e(b,c){var d=0;if(3==b.nodeType){var a=b.data.toUpperCase().indexOf(c);if(0<=a){d=document.createElement("span");d.className="highlight";a=b.splitText(a);a.splitText(c.length);var f=a.cloneNode(!0);d.appendChild(f);a.parentNode.replaceChild(d,a);d=1}}else if(1==b.nodeType&&b.childNodes&&!/(script|style)/i.test(b.tagName))for(a=0;a<b.childNodes.length;++a)a+=e(b.childNodes[a],c);return d}return this.length&&c&&c.length?this.each(function(){e(this,c.toUpperCase())}): this};jQuery.fn.removeHighlight=function(){return this.find("span.highlight").each(function(){this.parentNode.firstChild.nodeName;with(this.parentNode)replaceChild(this.firstChild,this),normalize()}).end()};
|
1
js/jquery.knob.min.js
vendored
Normal file
193
js/jquery.timeago.js
Normal file
@ -0,0 +1,193 @@
|
|||||||
|
/**
|
||||||
|
* Timeago is a jQuery plugin that makes it easy to support automatically
|
||||||
|
* updating fuzzy timestamps (e.g. "4 minutes ago" or "about 1 day ago").
|
||||||
|
*
|
||||||
|
* @name timeago
|
||||||
|
* @version 1.3.0
|
||||||
|
* @requires jQuery v1.2.3+
|
||||||
|
* @author Ryan McGeary
|
||||||
|
* @license MIT License - http://www.opensource.org/licenses/mit-license.php
|
||||||
|
*
|
||||||
|
* For usage and examples, visit:
|
||||||
|
* http://timeago.yarp.com/
|
||||||
|
*
|
||||||
|
* Copyright (c) 2008-2013, Ryan McGeary (ryan -[at]- mcgeary [*dot*] org)
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function (factory) {
|
||||||
|
if (typeof define === 'function' && define.amd) {
|
||||||
|
// AMD. Register as an anonymous module.
|
||||||
|
define(['jquery'], factory);
|
||||||
|
} else {
|
||||||
|
// Browser globals
|
||||||
|
factory(jQuery);
|
||||||
|
}
|
||||||
|
}(function ($) {
|
||||||
|
$.timeago = function(timestamp) {
|
||||||
|
if (timestamp instanceof Date) {
|
||||||
|
return inWords(timestamp);
|
||||||
|
} else if (typeof timestamp === "string") {
|
||||||
|
return inWords($.timeago.parse(timestamp));
|
||||||
|
} else if (typeof timestamp === "number") {
|
||||||
|
return inWords(new Date(timestamp));
|
||||||
|
} else {
|
||||||
|
return inWords($.timeago.datetime(timestamp));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
var $t = $.timeago;
|
||||||
|
|
||||||
|
$.extend($.timeago, {
|
||||||
|
settings: {
|
||||||
|
refreshMillis: 60000,
|
||||||
|
allowFuture: false,
|
||||||
|
localeTitle: false,
|
||||||
|
cutoff: 0,
|
||||||
|
strings: {
|
||||||
|
prefixAgo: null,
|
||||||
|
prefixFromNow: null,
|
||||||
|
suffixAgo: "ago",
|
||||||
|
suffixFromNow: "from now",
|
||||||
|
seconds: "less than a minute",
|
||||||
|
minute: "about a minute",
|
||||||
|
minutes: "%d minutes",
|
||||||
|
hour: "about an hour",
|
||||||
|
hours: "about %d hours",
|
||||||
|
day: "a day",
|
||||||
|
days: "%d days",
|
||||||
|
month: "about a month",
|
||||||
|
months: "%d months",
|
||||||
|
year: "about a year",
|
||||||
|
years: "%d years",
|
||||||
|
wordSeparator: " ",
|
||||||
|
numbers: []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
inWords: function(distanceMillis) {
|
||||||
|
var $l = this.settings.strings;
|
||||||
|
var prefix = $l.prefixAgo;
|
||||||
|
var suffix = $l.suffixAgo;
|
||||||
|
if (this.settings.allowFuture) {
|
||||||
|
if (distanceMillis < 0) {
|
||||||
|
prefix = $l.prefixFromNow;
|
||||||
|
suffix = $l.suffixFromNow;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var seconds = Math.abs(distanceMillis) / 1000;
|
||||||
|
var minutes = seconds / 60;
|
||||||
|
var hours = minutes / 60;
|
||||||
|
var days = hours / 24;
|
||||||
|
var years = days / 365;
|
||||||
|
|
||||||
|
function substitute(stringOrFunction, number) {
|
||||||
|
var string = $.isFunction(stringOrFunction) ? stringOrFunction(number, distanceMillis) : stringOrFunction;
|
||||||
|
var value = ($l.numbers && $l.numbers[number]) || number;
|
||||||
|
return string.replace(/%d/i, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
var words = seconds < 45 && substitute($l.seconds, Math.round(seconds)) ||
|
||||||
|
seconds < 90 && substitute($l.minute, 1) ||
|
||||||
|
minutes < 45 && substitute($l.minutes, Math.round(minutes)) ||
|
||||||
|
minutes < 90 && substitute($l.hour, 1) ||
|
||||||
|
hours < 24 && substitute($l.hours, Math.round(hours)) ||
|
||||||
|
hours < 42 && substitute($l.day, 1) ||
|
||||||
|
days < 30 && substitute($l.days, Math.round(days)) ||
|
||||||
|
days < 45 && substitute($l.month, 1) ||
|
||||||
|
days < 365 && substitute($l.months, Math.round(days / 30)) ||
|
||||||
|
years < 1.5 && substitute($l.year, 1) ||
|
||||||
|
substitute($l.years, Math.round(years));
|
||||||
|
|
||||||
|
var separator = $l.wordSeparator || "";
|
||||||
|
if ($l.wordSeparator === undefined) { separator = " "; }
|
||||||
|
return $.trim([prefix, words, suffix].join(separator));
|
||||||
|
},
|
||||||
|
parse: function(iso8601) {
|
||||||
|
var s = $.trim(iso8601);
|
||||||
|
s = s.replace(/\.\d+/,""); // remove milliseconds
|
||||||
|
s = s.replace(/-/,"/").replace(/-/,"/");
|
||||||
|
s = s.replace(/T/," ").replace(/Z/," UTC");
|
||||||
|
s = s.replace(/([\+\-]\d\d)\:?(\d\d)/," $1$2"); // -04:00 -> -0400
|
||||||
|
return new Date(s);
|
||||||
|
},
|
||||||
|
datetime: function(elem) {
|
||||||
|
var iso8601 = $t.isTime(elem) ? $(elem).attr("datetime") : $(elem).attr("title");
|
||||||
|
return $t.parse(iso8601);
|
||||||
|
},
|
||||||
|
isTime: function(elem) {
|
||||||
|
// jQuery's `is()` doesn't play well with HTML5 in IE
|
||||||
|
return $(elem).get(0).tagName.toLowerCase() === "time"; // $(elem).is("time");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// functions that can be called via $(el).timeago('action')
|
||||||
|
// init is default when no action is given
|
||||||
|
// functions are called with context of a single element
|
||||||
|
var functions = {
|
||||||
|
init: function(){
|
||||||
|
var refresh_el = $.proxy(refresh, this);
|
||||||
|
refresh_el();
|
||||||
|
var $s = $t.settings;
|
||||||
|
if ($s.refreshMillis > 0) {
|
||||||
|
setInterval(refresh_el, $s.refreshMillis);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
update: function(time){
|
||||||
|
$(this).data('timeago', { datetime: $t.parse(time) });
|
||||||
|
refresh.apply(this);
|
||||||
|
},
|
||||||
|
updateFromDOM: function(){
|
||||||
|
$(this).data('timeago', { datetime: $t.parse( $t.isTime(this) ? $(this).attr("datetime") : $(this).attr("title") ) });
|
||||||
|
refresh.apply(this);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$.fn.timeago = function(action, options) {
|
||||||
|
var fn = action ? functions[action] : functions.init;
|
||||||
|
if(!fn){
|
||||||
|
throw new Error("Unknown function name '"+ action +"' for timeago");
|
||||||
|
}
|
||||||
|
// each over objects here and call the requested function
|
||||||
|
this.each(function(){
|
||||||
|
fn.call(this, options);
|
||||||
|
});
|
||||||
|
return this;
|
||||||
|
};
|
||||||
|
|
||||||
|
function refresh() {
|
||||||
|
var data = prepareData(this);
|
||||||
|
var $s = $t.settings;
|
||||||
|
|
||||||
|
if (!isNaN(data.datetime)) {
|
||||||
|
if ( $s.cutoff == 0 || distance(data.datetime) < $s.cutoff) {
|
||||||
|
$(this).text(inWords(data.datetime));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareData(element) {
|
||||||
|
element = $(element);
|
||||||
|
if (!element.data("timeago")) {
|
||||||
|
element.data("timeago", { datetime: $t.datetime(element) });
|
||||||
|
var text = $.trim(element.text());
|
||||||
|
if ($t.settings.localeTitle) {
|
||||||
|
element.attr("title", element.data('timeago').datetime.toLocaleString());
|
||||||
|
} else if (text.length > 0 && !($t.isTime(element) && element.attr("title"))) {
|
||||||
|
element.attr("title", text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return element.data("timeago");
|
||||||
|
}
|
||||||
|
|
||||||
|
function inWords(date) {
|
||||||
|
return $t.inWords(distance(date));
|
||||||
|
}
|
||||||
|
|
||||||
|
function distance(date) {
|
||||||
|
return (new Date().getTime() - date.getTime());
|
||||||
|
}
|
||||||
|
|
||||||
|
// fix for IE6 suckage
|
||||||
|
document.createElement("abbr");
|
||||||
|
document.createElement("time");
|
||||||
|
}));
|
4
js/modernizr.js
Normal file
9523
js/wysihtml5-0.3.0.js
Normal file
1
license.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
Please see file "protected/docs/license.md" for licensing details.
|
1
protected/.htaccess
Normal file
@ -0,0 +1 @@
|
|||||||
|
deny from all
|
44
protected/behaviors/HActiveRecordBehavior.php
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HActiveRecordBehavior is attached to all active records.
|
||||||
|
*
|
||||||
|
* @package humhub.behaviors
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HActiveRecordBehavior extends CActiveRecordBehavior {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* On after construct event of an active record
|
||||||
|
*
|
||||||
|
* @param type $event
|
||||||
|
*/
|
||||||
|
public function afterConstruct($event) {
|
||||||
|
|
||||||
|
// Intercept this controller
|
||||||
|
Yii::app()->interceptor->intercept($this);
|
||||||
|
|
||||||
|
parent::afterConstruct($event);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
187
protected/behaviors/HContentAddonBehavior.php
Normal file
@ -0,0 +1,187 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HContentAddonBehavior should attached to every content addon model
|
||||||
|
*
|
||||||
|
* Items like Comments, Likes with linked to a
|
||||||
|
* HContentBehavior or HContentAddonBehavior
|
||||||
|
* Model
|
||||||
|
*
|
||||||
|
* An ContentAddon needs to have following DB Fields
|
||||||
|
* - id
|
||||||
|
* - object_model
|
||||||
|
* - object_id
|
||||||
|
*
|
||||||
|
* Which should point to SIContent or SIContentAddon
|
||||||
|
*
|
||||||
|
* @package humhub.behaviors
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HContentAddonBehavior extends HActiveRecordBehavior {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache Content Ooject and avoid multiple loading
|
||||||
|
*
|
||||||
|
* @var type
|
||||||
|
*/
|
||||||
|
private $_cacheContentObject = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an object which this Addon belongs to
|
||||||
|
* Only Support Content -> ContentAddon -> ContentAddon
|
||||||
|
*
|
||||||
|
* This is always a object which have the behavior: HContentBehavior
|
||||||
|
*
|
||||||
|
* e.g. Post / Like
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public function getContentObject() {
|
||||||
|
|
||||||
|
// Fastlane?
|
||||||
|
if ($this->_cacheContentObject != null)
|
||||||
|
return $this->_cacheContentObject;
|
||||||
|
|
||||||
|
$target = $this->getOwner()->getUnderlyingObject();
|
||||||
|
|
||||||
|
if ($target->asa('HContentBehavior') !== null) {
|
||||||
|
$this->_cacheContentObject = $target;
|
||||||
|
return $target;
|
||||||
|
} elseif ($target->asa('HContentAddonBehavior') !== null) {
|
||||||
|
$target2 = $target->getOwner()->getUnderlyingObject();
|
||||||
|
|
||||||
|
if ($target2->asa('HContentBehavior') !== null) {
|
||||||
|
$this->_cacheContentObject = $target2;
|
||||||
|
return $target2;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Maybe Multiple Nested?
|
||||||
|
}
|
||||||
|
throw new CHttpException(500, Yii::t('base', 'Invalid content object!'));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the given / or current user can delete this content.
|
||||||
|
* Currently only the creator can remove.
|
||||||
|
*
|
||||||
|
* @param type $userId
|
||||||
|
*/
|
||||||
|
public function canDelete($userId = "") {
|
||||||
|
if ($userId == "")
|
||||||
|
$userId = Yii::app()->user->id;
|
||||||
|
|
||||||
|
if ($this->getOwner()->created_by == $userId)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deligate to the underlying content object
|
||||||
|
*
|
||||||
|
* @param type $userId
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function canRead($userId = "") {
|
||||||
|
return $this->getContentObject()->canRead($userId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if this content can be changed
|
||||||
|
*
|
||||||
|
* @param type $userId
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function canWrite($userId = "") {
|
||||||
|
if ($userId == "")
|
||||||
|
$userId = Yii::app()->user->id;
|
||||||
|
|
||||||
|
if ($this->getOwner()->created_by == $userId)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns textual title for this content addon
|
||||||
|
*
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function getContentTitle() {
|
||||||
|
$objectModel = get_class($this->getOwner()); // e.g. Post
|
||||||
|
|
||||||
|
return $objectModel . " (" . $this->getOwner()->id . ")";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Before saving, check target on content addon
|
||||||
|
*/
|
||||||
|
public function beforeSave($event) {
|
||||||
|
|
||||||
|
if ($this->getOwner()->getUnderlyingObject() == null)
|
||||||
|
throw new CHttpException(500, 'Could not load comment target!');
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After saving
|
||||||
|
*/
|
||||||
|
public function afterSave($event) {
|
||||||
|
// Also load corresponding SIContent Object and change "updated_at & updated_by" time
|
||||||
|
$this->getContentObject()->save();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After delete
|
||||||
|
*/
|
||||||
|
public function afterDelete($event) {
|
||||||
|
// Also load corresponding SIContent Object and change "updated_at & updated_by" time
|
||||||
|
$this->getContentObject()->save();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Before deleting a SIContent try to delete all corresponding SIContentAddons.
|
||||||
|
*/
|
||||||
|
public function beforeDelete($event) {
|
||||||
|
|
||||||
|
$objectModel = get_class($this->getOwner()); // e.g. Comment
|
||||||
|
// delete all comments (sub comments?) - not implemented yet :-)
|
||||||
|
$comments = Comment::model()->findAllByAttributes(array('object_model' => $objectModel, 'object_id' => $this->getOwner()->id));
|
||||||
|
foreach ($comments as $comment) {
|
||||||
|
$comment->delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
// delete all likes
|
||||||
|
$likes = Like::model()->findAllByAttributes(array('object_model' => $objectModel, 'object_id' => $this->getOwner()->id));
|
||||||
|
foreach ($likes as $like) {
|
||||||
|
$like->delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
// delete all activities
|
||||||
|
$activities = Activity::model()->findAllByAttributes(array('object_model' => $objectModel, 'object_id' => $this->getOwner()->id));
|
||||||
|
foreach ($activities as $activity) {
|
||||||
|
$activity->delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
57
protected/behaviors/HContentBaseBehavior.php
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The SIContentBaseBahaviour is attached to all models which act as contentbase.
|
||||||
|
*
|
||||||
|
* A ContentBase can be Space or User. Essentially a ContentBase holds a
|
||||||
|
* bunch of Content Objects.
|
||||||
|
*
|
||||||
|
* Each ContentBase should own a Wall Object.
|
||||||
|
* A Content Base needs a GUID Database Field
|
||||||
|
*
|
||||||
|
* Additionally each ContentBase has a ProfileImage.
|
||||||
|
*
|
||||||
|
* Ideas/ToDo:
|
||||||
|
* - GetUrl (Instead of Profile Url)
|
||||||
|
*
|
||||||
|
* @package humhub.behaviors
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HContentBaseBehavior extends HActiveRecordBehavior {
|
||||||
|
// Auto Add Wall?
|
||||||
|
// Auto Delete Wall?
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the Profile Image Object for this Content Base
|
||||||
|
*
|
||||||
|
* @return ProfileImage
|
||||||
|
*/
|
||||||
|
public function getProfileImage() {
|
||||||
|
|
||||||
|
if (get_class($this->getOwner()) == 'Space') {
|
||||||
|
return new ProfileImage($this->getOwner()->guid, 'default_workspace');
|
||||||
|
}
|
||||||
|
return new ProfileImage($this->getOwner()->guid);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
141
protected/behaviors/HContentBehavior.php
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Behavior for Contents Objects
|
||||||
|
*
|
||||||
|
* Generally binds a Content Object to the given ActiveRecord
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.behaviors
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HContentBehavior extends HActiveRecordBehavior {
|
||||||
|
|
||||||
|
private $_content = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Before deleting of a content object (e.g. post, activity, ...)
|
||||||
|
*
|
||||||
|
* @param type $event
|
||||||
|
*/
|
||||||
|
public function beforeDelete($event) {
|
||||||
|
parent::beforeDelete($event);
|
||||||
|
|
||||||
|
// Make sure corresponding content object is loaded
|
||||||
|
// So we can delete is afterwards
|
||||||
|
$this->getContentMeta();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After deleting of a content object (e.g. post, activity, ...)
|
||||||
|
*
|
||||||
|
* @param type $event
|
||||||
|
*/
|
||||||
|
public function afterDelete($event) {
|
||||||
|
parent::afterDelete($event);
|
||||||
|
|
||||||
|
// Search Stuff
|
||||||
|
if ($this->getOwner() instanceof ISearchable) {
|
||||||
|
|
||||||
|
if (!$this->getOwner()->isNewRecord)
|
||||||
|
HSearch::getInstance()->deleteModel($this->getOwner());
|
||||||
|
|
||||||
|
HSearch::getInstance()->addModel($this->getOwner());
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->getContentMeta()->delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After saving of a content object (e.g. post, activity, ...)
|
||||||
|
*
|
||||||
|
* @param type $event
|
||||||
|
*/
|
||||||
|
public function afterSave($event) {
|
||||||
|
|
||||||
|
parent::afterSave($event);
|
||||||
|
|
||||||
|
if ($this->getContentMeta()->isNewRecord) {
|
||||||
|
$this->getContentMeta()->object_model = get_class($this->getOwner());
|
||||||
|
$this->getContentMeta()->object_id = $this->getOwner()->id;
|
||||||
|
|
||||||
|
// Because Content Model dont set this automatically atm
|
||||||
|
// For update reasons
|
||||||
|
// On later release remove this, and change Content Model to set this automatically
|
||||||
|
$this->getContentMeta()->created_at = $this->getOwner()->created_at;
|
||||||
|
$this->getContentMeta()->created_by = $this->getOwner()->created_by;
|
||||||
|
$this->getContentMeta()->updated_at = $this->getOwner()->updated_at;
|
||||||
|
$this->getContentMeta()->updated_by = $this->getOwner()->updated_by;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->getContentMeta()->save();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After finding a record
|
||||||
|
*
|
||||||
|
* @param type $event
|
||||||
|
*/
|
||||||
|
public function afterFind($event) {
|
||||||
|
parent::afterFind($event);
|
||||||
|
|
||||||
|
// Make sure corresponding content object is empty
|
||||||
|
$this->_content = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns corresponding Content (Meta) Object for underlying content class.
|
||||||
|
*
|
||||||
|
* ToDo: After Find Remove Existing ContentMeta Object
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function getContentMeta() {
|
||||||
|
|
||||||
|
// Content Object already instanciated
|
||||||
|
if ($this->_content != null)
|
||||||
|
return $this->_content;
|
||||||
|
|
||||||
|
// Is new record, so return empty content object
|
||||||
|
if ($this->getOwner()->isNewRecord) {
|
||||||
|
$this->_content = new Content;
|
||||||
|
|
||||||
|
// some defaults
|
||||||
|
$this->_content->sticked = 0;
|
||||||
|
$this->_content->archived = 0;
|
||||||
|
$this->_content->visibility = Content::VISIBILITY_PRIVATE;
|
||||||
|
|
||||||
|
return $this->_content;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->_content = Content::model()->findByAttributes(array('object_model' => get_class($this->getOwner()), 'object_id' => $this->getOwner()->id));
|
||||||
|
|
||||||
|
if ($this->_content == null) {
|
||||||
|
throw new CHttpException(500, Yii::t('base', 'Missing underlying content object!'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->_content;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
98
protected/behaviors/HUnderlyingObjectBehavior.php
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A SIUnderlyingObjectBahavior adds the ability to models/active records
|
||||||
|
* to link to any other model/active record.
|
||||||
|
*
|
||||||
|
* This is archived by the database fields object_model & object_id.
|
||||||
|
*
|
||||||
|
* Required database fields:
|
||||||
|
* - object_model
|
||||||
|
* - object_id
|
||||||
|
*
|
||||||
|
* @package humhub.behaviors
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HUnderlyingObjectBehavior extends HActiveRecordBehavior {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The underlying object needs to be a "instanceof" at least one
|
||||||
|
* of this values.
|
||||||
|
*
|
||||||
|
* (Its also possible to specify a CBehavior name)
|
||||||
|
*
|
||||||
|
* @var type
|
||||||
|
*/
|
||||||
|
public $mustBeInstanceOf = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache Object to avoid multiple loading
|
||||||
|
*
|
||||||
|
* @var type
|
||||||
|
*/
|
||||||
|
private $_cachedObject = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache Object is null
|
||||||
|
*/
|
||||||
|
private $_cached = false;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Returns the Underlying Object
|
||||||
|
*
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
|
||||||
|
public function getUnderlyingObject() {
|
||||||
|
|
||||||
|
if ($this->_cached)
|
||||||
|
return $this->_cachedObject;
|
||||||
|
|
||||||
|
$className = $this->getOwner()->object_model;
|
||||||
|
|
||||||
|
if ($className == "") {
|
||||||
|
$this->_cached = true;
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$object = $className::model()->findByPk($this->getOwner()->object_id);
|
||||||
|
|
||||||
|
if (count($this->mustBeInstanceOf) == 0 || $object == null) {
|
||||||
|
$this->_cached = true;
|
||||||
|
$this->_cachedObject = $object;
|
||||||
|
return $object;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validates object
|
||||||
|
foreach ($this->mustBeInstanceOf as $instance) {
|
||||||
|
if ($object instanceof $instance || $object->asa($instance) !== null) {
|
||||||
|
$this->_cached = true;
|
||||||
|
$this->_cachedObject = $object;
|
||||||
|
return $object;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new CHttpException(500, 'Underlying object of invalid type! (' . $className . ')');
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
212
protected/commands/shell/EMailing/EMailing.php
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The E-Mailing command sends daily or hourly e-mails updates with current
|
||||||
|
* pending activities and notification.
|
||||||
|
*
|
||||||
|
* @package humhub.commands.shell.EMailing
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class EMailing extends HConsoleCommand {
|
||||||
|
|
||||||
|
private $mode = "hourly"; // daily
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run method for EMailing System
|
||||||
|
*
|
||||||
|
* @param type $args
|
||||||
|
*/
|
||||||
|
|
||||||
|
public function run($args) {
|
||||||
|
|
||||||
|
$this->printHeader('E-Mail Interface');
|
||||||
|
|
||||||
|
if (!isset($args[0]) || ($args[0] != "daily" && $args[0] != 'hourly')) {
|
||||||
|
print "\n Run with parameter:\n" .
|
||||||
|
"\t daily - for Daily Mailings\n" .
|
||||||
|
"\t hourly - for Hourly Mailings\n";
|
||||||
|
print "\n\n";
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
$this->mode = $args[0];
|
||||||
|
Yii::import("application.modules_core.wall.*", true);
|
||||||
|
|
||||||
|
Yii::app()->request->setBaseUrl(HSetting::Get('baseUrl'));
|
||||||
|
|
||||||
|
$users = User::model()->with('httpSessions')->findAll();
|
||||||
|
|
||||||
|
foreach ($users as $user) {
|
||||||
|
|
||||||
|
print "Processing : " . $user->email . ": ";
|
||||||
|
|
||||||
|
$user->checkWall();
|
||||||
|
|
||||||
|
$notificationContent = $this->getNotificationContent($user);
|
||||||
|
$activityContent = $this->getActivityContent($user);
|
||||||
|
|
||||||
|
// Something new?
|
||||||
|
if ($notificationContent == "" && $activityContent == "") {
|
||||||
|
print "Nothing new! \n";
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$message = new HMailMessage();
|
||||||
|
$message->view = 'application.views.mail.EMailing';
|
||||||
|
$message->addFrom(HSetting::Get('systemEmailAddress', 'mailing'), HSetting::Get('systemEmailName', 'mailing'));
|
||||||
|
$message->addTo($user->email);
|
||||||
|
|
||||||
|
if ($this->mode == 'hourly') {
|
||||||
|
$message->subject = Yii::t('base', "Latest news");
|
||||||
|
} else {
|
||||||
|
$message->subject = Yii::t('base', "Your daily summary");
|
||||||
|
}
|
||||||
|
|
||||||
|
$message->setBody(array(
|
||||||
|
'notificationContent' => $notificationContent,
|
||||||
|
'activityContent' => $activityContent,
|
||||||
|
'user' => $user,
|
||||||
|
), 'text/html');
|
||||||
|
Yii::app()->mail->send($message);
|
||||||
|
|
||||||
|
print "Sent! \n";
|
||||||
|
}
|
||||||
|
|
||||||
|
print "\nEMailing completed.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns notification content by given user.
|
||||||
|
*
|
||||||
|
* The output will generated on current mode.
|
||||||
|
*
|
||||||
|
* @param type $user
|
||||||
|
* @return string email output
|
||||||
|
*/
|
||||||
|
private function getNotificationContent($user) {
|
||||||
|
|
||||||
|
// Never receive notifications
|
||||||
|
if ($user->receive_email_notifications == User::RECEIVE_EMAIL_NEVER) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// We are in hourly mode and user wants daily
|
||||||
|
if ($this->mode == 'hourly' && $user->receive_email_notifications == User::RECEIVE_EMAIL_DAILY_SUMMARY) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// We are in daily mode and user dont wants daily reports
|
||||||
|
if ($this->mode == 'daily' && $user->receive_email_notifications != User::RECEIVE_EMAIL_DAILY_SUMMARY) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// User wants only when offline and is online
|
||||||
|
if ($this->mode == 'hourly') {
|
||||||
|
$isOnline = (count($user->httpSessions) > 0);
|
||||||
|
if ($user->receive_email_notifications == User::RECEIVE_EMAIL_WHEN_OFFLINE && $isOnline) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get not seen notifcation, order by created_at
|
||||||
|
$criteria = new CDbCriteria();
|
||||||
|
$criteria->order = 'created_at DESC';
|
||||||
|
$notifications = Notification::model()->findAllByAttributes(array('user_id' => $user->id, 'seen' => 0, 'emailed' => 0), $criteria);
|
||||||
|
|
||||||
|
// Nothin new
|
||||||
|
if (count($notifications) == 0) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate notification output
|
||||||
|
$output = "";
|
||||||
|
foreach ($notifications as $notification) {
|
||||||
|
$output .= $notification->getMailOut();
|
||||||
|
$notification->emailed = 1;
|
||||||
|
$notification->save();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return activity content by given user.
|
||||||
|
*
|
||||||
|
* This output is generated by current mode.
|
||||||
|
*
|
||||||
|
* @param type $user
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private function getActivityContent($user) {
|
||||||
|
|
||||||
|
// User never wants activity content
|
||||||
|
if ($user->receive_email_activities == User::RECEIVE_EMAIL_NEVER) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// We are in hourly mode and user wants receive a daily summary
|
||||||
|
if ($this->mode == 'hourly' && $user->receive_email_activities == User::RECEIVE_EMAIL_DAILY_SUMMARY) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// We are in daily mode and user wants receive not daily
|
||||||
|
if ($this->mode == 'daily' && $user->receive_email_activities != User::RECEIVE_EMAIL_DAILY_SUMMARY) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// User is online and want only receive when offline
|
||||||
|
if ($this->mode == 'hourly') {
|
||||||
|
$isOnline = (count($user->httpSessions) > 0);
|
||||||
|
if ($user->receive_email_activities == User::RECEIVE_EMAIL_WHEN_OFFLINE && $isOnline) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
$lastMailDate = $user->last_activity_email;
|
||||||
|
if ($lastMailDate == "" || $lastMailDate == "0000-00-00 00:00:00") {
|
||||||
|
$lastMailDate = new CDbExpression('NOW() - INTERVAL 24 HOUR');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Get Stream contents
|
||||||
|
$action = new StreamAction(null, 'console');
|
||||||
|
$action->mode = 'activity';
|
||||||
|
$action->type = 'dashboard';
|
||||||
|
$action->userId = $user->id;
|
||||||
|
$action->userWallId = $user->wall_id;
|
||||||
|
$action->wallEntryLimit = 50;
|
||||||
|
$action->wallEntryDateTo = $lastMailDate;
|
||||||
|
$activities = $action->runConsole();
|
||||||
|
|
||||||
|
# Save last run
|
||||||
|
$user->last_activity_email = new CDbExpression('NOW()');
|
||||||
|
$user->save();
|
||||||
|
|
||||||
|
// Nothin new
|
||||||
|
if ($activities['counter'] == 0) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return Output
|
||||||
|
return $activities['output'];
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
56
protected/commands/shell/EMailing/TestMail.php
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The TestMail command is used to test the e-mail subsystem.
|
||||||
|
*
|
||||||
|
* @package humhub.commands.shell.EMailing
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class TestMail extends HConsoleCommand {
|
||||||
|
|
||||||
|
public function run($args) {
|
||||||
|
print " E-Mail - Test Interface\n";
|
||||||
|
print "-------------------------------------------------------------------------\n\n";
|
||||||
|
|
||||||
|
if (!isset($args[0]) || ($args[0] == "")) {
|
||||||
|
print "\n Run with parameter [email]!\n" .
|
||||||
|
print "\n\n";
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
$email = $args[0];
|
||||||
|
|
||||||
|
$user = User::model()->findByPk(1);
|
||||||
|
|
||||||
|
$message = new HMailMessage();
|
||||||
|
$message->view = 'application.views.mail.EMailing';
|
||||||
|
$message->addFrom(HSetting::Get('systemEmailAddress', 'mailing'), HSetting::Get('systemEmailName', 'mailing'));
|
||||||
|
$message->addTo($email);
|
||||||
|
$message->subject = "Test Mail";
|
||||||
|
|
||||||
|
$message->setBody(array('user' => $user), 'text/html');
|
||||||
|
Yii::app()->mail->send($message);
|
||||||
|
|
||||||
|
print "Sent! \n";
|
||||||
|
|
||||||
|
print "\nEMailing completed.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
95
protected/commands/shell/Maintain/IntegrityChecker.php
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* IntegrityChecker command validates the current database and tries to fix
|
||||||
|
* currupted records.
|
||||||
|
*
|
||||||
|
* Also third party modules can add checking methods by intercepting the run
|
||||||
|
* event.
|
||||||
|
*
|
||||||
|
* @package humhub.commands.shell.Maintain
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class IntegrityChecker extends HConsoleCommand {
|
||||||
|
|
||||||
|
public $simulate = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runs Integrity Checker
|
||||||
|
*
|
||||||
|
* @param type $args
|
||||||
|
*/
|
||||||
|
public function run($args) {
|
||||||
|
|
||||||
|
$this->printHeader('Integrity Checker');
|
||||||
|
|
||||||
|
if ($this->simulate) {
|
||||||
|
print "Simulation Mode!\n\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->doBaseTasks();
|
||||||
|
|
||||||
|
if ($this->hasEventHandler('onRun'))
|
||||||
|
$this->onRun(new CEvent($this));
|
||||||
|
|
||||||
|
print "\n\n";
|
||||||
|
print "Finished! Integrity Check done!\n\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Do general tasks used application whide
|
||||||
|
*/
|
||||||
|
protected function doBaseTasks() {
|
||||||
|
|
||||||
|
$this->showTestHeadline("Checking application base structure");
|
||||||
|
|
||||||
|
if (HSetting::Get('secret') == "" || HSetting::Get('secret') == null) {
|
||||||
|
HSetting::Set('secret', UUID::v4());
|
||||||
|
$this->showFix('Setting missing application secret!');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* onRun Event is used to notify modules.
|
||||||
|
*
|
||||||
|
* @param type $event
|
||||||
|
*/
|
||||||
|
public function onRun($event) {
|
||||||
|
$this->raiseEvent('onRun', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Shows a headline in console
|
||||||
|
* @param type $title
|
||||||
|
*/
|
||||||
|
public function showTestHeadline($title) {
|
||||||
|
print "*** " . $title . "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use this method to show a fix
|
||||||
|
* @param type $msg
|
||||||
|
*/
|
||||||
|
public function showFix($msg) {
|
||||||
|
print "\t" . $msg . "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
42
protected/commands/shell/SearchIndexer/Optimize.php
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optimize command optimizing the search index by calling underlying
|
||||||
|
* Zend Lucence Index optimizer.
|
||||||
|
*
|
||||||
|
* @package humhub.commands.shell.SearchIndexer
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class Optimize extends HConsoleCommand {
|
||||||
|
|
||||||
|
public function run($args) {
|
||||||
|
|
||||||
|
$this->printHeader('Search Index Optimizer');
|
||||||
|
|
||||||
|
|
||||||
|
print "- Optimizing ...\n";
|
||||||
|
HSearch::getInstance()->optimize();
|
||||||
|
print "- Search Index successfully optimized!\n";
|
||||||
|
|
||||||
|
print "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
44
protected/commands/shell/SearchIndexer/Rebuilder.php
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rebuilder command purges all search index files and rebuilds the whole index.
|
||||||
|
*
|
||||||
|
* @package humhub.commands.shell.SearchIndexer
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class Rebuilder extends HConsoleCommand {
|
||||||
|
|
||||||
|
public function run($args) {
|
||||||
|
|
||||||
|
$this->printHeader('Rebuild Search Index\n');
|
||||||
|
|
||||||
|
print "Deleting old index files: ";
|
||||||
|
HSearch::getInstance()->flushIndex();
|
||||||
|
print " done!\n";
|
||||||
|
|
||||||
|
print "Rebuilding index: ";
|
||||||
|
HSearch::getInstance()->rebuild();
|
||||||
|
print " done!\n";
|
||||||
|
|
||||||
|
print "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
120
protected/commands/shell/ZCron/ZCronRunner.php
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ZCronRunner command is a generic cron interface.
|
||||||
|
*
|
||||||
|
* Third party modules can intercept run events to attach methods to the cron
|
||||||
|
* subsystem.
|
||||||
|
*
|
||||||
|
* @package humhub.commands.shell.ZCron
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class ZCronRunner extends HConsoleCommand {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cron Intervals
|
||||||
|
*/
|
||||||
|
const INTERVAL_DAILY = 1;
|
||||||
|
const INTERVAL_HOURLY = 2;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var integer current interval
|
||||||
|
*/
|
||||||
|
public $interval = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run method for CronRunner
|
||||||
|
*
|
||||||
|
* @param type $args
|
||||||
|
*/
|
||||||
|
public function run($args) {
|
||||||
|
|
||||||
|
$this->printHeader('Cron Interface');
|
||||||
|
|
||||||
|
if (!isset($args[0]) || ($args[0] != "daily" && $args[0] != 'hourly')) {
|
||||||
|
print "\n Run with parameter:\n" .
|
||||||
|
"\t daily - for Daily Interval\n" .
|
||||||
|
"\t hourly - for Hourly Interval \n";
|
||||||
|
print "\n\n";
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set current interval to attribute
|
||||||
|
if ($args[0] == 'daily')
|
||||||
|
$this->interval = self::INTERVAL_DAILY;
|
||||||
|
elseif ($args[0] == 'hourly')
|
||||||
|
$this->interval = self::INTERVAL_HOURLY;
|
||||||
|
else
|
||||||
|
throw new CException(500, 'Invalid cron interval!');
|
||||||
|
|
||||||
|
// Do some base tasks, handle by event in future
|
||||||
|
if ($this->interval == self::INTERVAL_HOURLY) {
|
||||||
|
|
||||||
|
// Raise Event for Module specific tasks
|
||||||
|
if ($this->hasEventHandler('onHourlyRun'))
|
||||||
|
$this->onHourlyRun(new CEvent($this));
|
||||||
|
|
||||||
|
print "- Optimizing Search Index\n";
|
||||||
|
// Optimize Search Index
|
||||||
|
HSearch::getInstance()->Optimize();
|
||||||
|
|
||||||
|
print "- Invoking EMailing hourly\n\n";
|
||||||
|
// Execute Hourly Mailing Runner
|
||||||
|
Yii::import('application.commands.shell.EMailing.*');
|
||||||
|
$command = new EMailing('test', 'test');
|
||||||
|
$command->run(array('hourly'));
|
||||||
|
|
||||||
|
HSetting::Set('cronLastHourlyRun', time());
|
||||||
|
} elseif ($this->interval == self::INTERVAL_DAILY) {
|
||||||
|
// Raise Event for Module specific tasks
|
||||||
|
|
||||||
|
if ($this->hasEventHandler('onDailyRun'))
|
||||||
|
$this->onDailyRun(new CEvent($this));
|
||||||
|
|
||||||
|
// Execute Daily Mailing Runner
|
||||||
|
print "- Invoking EMailing daily\n\n";
|
||||||
|
Yii::import('application.commands.shell.EMailing.*');
|
||||||
|
$command = new EMailing('test', 'test');
|
||||||
|
$command->run(array('daily'));
|
||||||
|
|
||||||
|
HSetting::Set('cronLastDailyRun', time());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This event is raised after the cron hourly run is performed.
|
||||||
|
*
|
||||||
|
* @param CEvent $event the event parameter
|
||||||
|
*/
|
||||||
|
public function onHourlyRun($event) {
|
||||||
|
$this->raiseEvent('onHourlyRun', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This event is raised after the cron hourly run is performed.
|
||||||
|
*
|
||||||
|
* @param CEvent $event the event parameter
|
||||||
|
*/
|
||||||
|
public function onDailyRun($event) {
|
||||||
|
$this->raiseEvent('onDailyRun', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
91
protected/commands/shell/ZDbMigration.php
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ZDbMigration extends EDbMigration with better interactive support.
|
||||||
|
*
|
||||||
|
* @package humhub.commands
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class ZDbMigration extends CDbMigration {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var EMigrateCommand
|
||||||
|
*/
|
||||||
|
private $migrateCommand;
|
||||||
|
protected $interactive = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param EMigrateCommand $migrateCommand
|
||||||
|
*/
|
||||||
|
public function setCommand($migrateCommand) {
|
||||||
|
$this->migrateCommand = $migrateCommand;
|
||||||
|
$this->interactive = $migrateCommand->interactive;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see CConsoleCommand::confirm()
|
||||||
|
* @param string $message
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function confirm($message) {
|
||||||
|
if (!$this->interactive) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return $this->migrateCommand->confirm($message);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see CConsoleCommand::prompt()
|
||||||
|
* @param string $message
|
||||||
|
* @param mixed $defaultValue will be returned when interactive is false
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function prompt($message, $defaultValue) {
|
||||||
|
if (!$this->interactive) {
|
||||||
|
return $defaultValue;
|
||||||
|
}
|
||||||
|
return $this->migrateCommand->prompt($message);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Executes a SQL statement. Silently. (only show sql on exception)
|
||||||
|
* This method executes the specified SQL statement using {@link dbConnection}.
|
||||||
|
* @param string $sql the SQL statement to be executed
|
||||||
|
* @param array $params input parameters (name=>value) for the SQL execution. See {@link CDbCommand::execute} for more details.
|
||||||
|
* @param boolean $verbose
|
||||||
|
*/
|
||||||
|
public function execute($sql, $params = array(), $verbose = true) {
|
||||||
|
if ($verbose) {
|
||||||
|
parent::execute($sql, $params);
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
echo " > execute SQL ...";
|
||||||
|
$time = microtime(true);
|
||||||
|
$this->getDbConnection()->createCommand($sql)->execute($params);
|
||||||
|
echo " done (time: " . sprintf('%.3f', microtime(true) - $time) . "s)\n";
|
||||||
|
} catch (CException $e) {
|
||||||
|
echo " failed.\n\n";
|
||||||
|
throw $e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
149
protected/commands/shell/ZMessageCommand.php
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
Yii::import('application.vendors.yii.cli.commands.shell.*');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ZMessageCommand extends Yiis MessageCommand with better module support.
|
||||||
|
*
|
||||||
|
* - Also add module messages into module/messages dir.
|
||||||
|
*
|
||||||
|
* @package humhub.commands
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class ZMessageCommand extends MessageCommand {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the action.
|
||||||
|
* @param array command line parameters specific for this command
|
||||||
|
*/
|
||||||
|
public function run($args) {
|
||||||
|
|
||||||
|
if (!isset($args[0]))
|
||||||
|
$this->usageError('the configuration file is not specified.');
|
||||||
|
if (!is_file($args[0]))
|
||||||
|
$this->usageError("the configuration file {$args[0]} does not exist.");
|
||||||
|
|
||||||
|
$config = require_once($args[0]);
|
||||||
|
$translator = 'Yii::t';
|
||||||
|
extract($config);
|
||||||
|
|
||||||
|
if (!isset($sourcePath, $messagePath, $languages))
|
||||||
|
$this->usageError('The configuration file must specify "sourcePath", "messagePath" and "languages".');
|
||||||
|
if (!is_dir($sourcePath))
|
||||||
|
$this->usageError("The source path $sourcePath is not a valid directory.");
|
||||||
|
if (!is_dir($messagePath))
|
||||||
|
$this->usageError("The message path $messagePath is not a valid directory.");
|
||||||
|
if (empty($languages))
|
||||||
|
$this->usageError("Languages cannot be empty.");
|
||||||
|
|
||||||
|
if (!isset($overwrite))
|
||||||
|
$overwrite = false;
|
||||||
|
|
||||||
|
if (!isset($removeOld))
|
||||||
|
$removeOld = false;
|
||||||
|
|
||||||
|
if (!isset($sort))
|
||||||
|
$sort = false;
|
||||||
|
|
||||||
|
$options = array();
|
||||||
|
if (isset($fileTypes))
|
||||||
|
$options['fileTypes'] = $fileTypes;
|
||||||
|
if (isset($exclude))
|
||||||
|
$options['exclude'] = $exclude;
|
||||||
|
$files = CFileHelper::findFiles(realpath($sourcePath), $options);
|
||||||
|
|
||||||
|
$messages = array();
|
||||||
|
foreach ($files as $file)
|
||||||
|
$messages = array_merge_recursive($messages, $this->extractMessages($file, $translator));
|
||||||
|
|
||||||
|
|
||||||
|
foreach ($messages as $category => $msgs) {
|
||||||
|
|
||||||
|
$dir = "";
|
||||||
|
if (($pos = strpos($category, '.')) !== false) {
|
||||||
|
$parts = explode(".", $category);
|
||||||
|
$className = $parts[0];
|
||||||
|
|
||||||
|
if (!$this->isModuleClassDefined($className)) {
|
||||||
|
print "Skip not enabled module: " . $className . "\n";
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get base Path of Module
|
||||||
|
$class = new ReflectionClass($className);
|
||||||
|
$dir = dirname($class->getFileName()) . DIRECTORY_SEPARATOR . 'messages' . DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
$category = $parts[1];
|
||||||
|
} else {
|
||||||
|
$dir = $messagePath . DIRECTORY_SEPARATOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip when there is no message dir
|
||||||
|
if (!is_dir($dir)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($languages as $language) {
|
||||||
|
$messageDir = $dir . DIRECTORY_SEPARATOR . $language;
|
||||||
|
if (!is_dir($messageDir))
|
||||||
|
@mkdir($messageDir);
|
||||||
|
|
||||||
|
$msgs = array_values(array_unique($msgs));
|
||||||
|
$this->generateMessageFile($msgs, $messageDir . DIRECTORY_SEPARATOR . $category . '.php', $overwrite, $removeOld, $sort);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function extractMessages($fileName, $translator) {
|
||||||
|
echo "Extracting messages from $fileName...\n";
|
||||||
|
$subject = file_get_contents($fileName);
|
||||||
|
$messages = array();
|
||||||
|
if (!is_array($translator))
|
||||||
|
$translator = array($translator);
|
||||||
|
|
||||||
|
foreach ($translator as $currentTranslator) {
|
||||||
|
$n = preg_match_all('/\b' . $currentTranslator . '\s*\(\s*(\'[\w.]*?(?<!\.)\'|"[\w.]*?(?<!\.)")\s*,\s*(\'.*?(?<!\\\\)\'|".*?(?<!\\\\)")\s*[,\)]/s', $subject, $matches, PREG_SET_ORDER);
|
||||||
|
|
||||||
|
for ($i = 0; $i < $n; ++$i) {
|
||||||
|
$category = substr($matches[$i][1], 1, -1);
|
||||||
|
$message = $matches[$i][2];
|
||||||
|
$messages[$category][] = eval("return $message;"); // use eval to eliminate quote escape
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $messages;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if given module is registered to the yii application.
|
||||||
|
* This is needed to prevent class autoload errrors when module/class is not enabled.
|
||||||
|
*
|
||||||
|
* @param type $className
|
||||||
|
* @return boolean
|
||||||
|
*/
|
||||||
|
public function isModuleClassDefined($className) {
|
||||||
|
foreach (Yii::app()->modules as $mod) {
|
||||||
|
if (strpos($mod['class'], $className) !== false) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
94
protected/commands/shell/ZMigrateCommand.php
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'ZDbMigration.php');
|
||||||
|
Yii::import('application.extensions.migrate-command.EMigrateCommand');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ZMigrateCommand extends EMigrateCommand with better interactive support.
|
||||||
|
*
|
||||||
|
* @package humhub.commands
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class ZMigrateCommand extends EMigrateCommand {
|
||||||
|
|
||||||
|
public $migrationTable = 'migration';
|
||||||
|
|
||||||
|
public static function AutoMigrate() {
|
||||||
|
|
||||||
|
|
||||||
|
/** $commandPath = Yii::app()->getBasePath() . DIRECTORY_SEPARATOR . 'commands';
|
||||||
|
$runner->addCommands($commandPath);
|
||||||
|
$commandPath = Yii::getFrameworkPath() . DIRECTORY_SEPARATOR . 'cli' . DIRECTORY_SEPARATOR . 'commands';
|
||||||
|
$runner->addCommands($commandPath); */
|
||||||
|
$runner = new CConsoleCommandRunner();
|
||||||
|
|
||||||
|
$runner->commands = array(
|
||||||
|
'migrate' => array(
|
||||||
|
'class' => 'applications.commands.shell.ZMigrateCommand',
|
||||||
|
'interactive' => false,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
$args = array('yiic', 'migrate', '--interactive=0');
|
||||||
|
ob_start();
|
||||||
|
$runner->run($args);
|
||||||
|
return htmlentities(ob_get_clean(), null, Yii::app()->charset);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function init() {
|
||||||
|
print "Autodetecting Modules....\n";
|
||||||
|
|
||||||
|
$modulePaths = array();
|
||||||
|
foreach (Yii::app()->moduleManager->getRegisteredModules() as $moduleId => $moduleInfo) {
|
||||||
|
|
||||||
|
// Convert path.to.example.ExampleModule to path.to.example.migrations
|
||||||
|
$path = explode(".", $moduleInfo['class']);
|
||||||
|
array_pop($path);
|
||||||
|
$path[] = $this->migrationSubPath;
|
||||||
|
$migrationPath = implode(".", $path);
|
||||||
|
|
||||||
|
if (is_dir(Yii::getPathOfAlias($migrationPath))) {
|
||||||
|
$modulePaths[$moduleId] = $migrationPath;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->modulePaths = $modulePaths;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function instantiateMigration($class) {
|
||||||
|
|
||||||
|
$migration = new $class;
|
||||||
|
$migration->setDbConnection($this->getDbConnection());
|
||||||
|
if ($migration instanceof EDbMigration) {
|
||||||
|
/** @var EDbMigration $migration */
|
||||||
|
$migration->setCommand($this);
|
||||||
|
}
|
||||||
|
return $migration;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getTemplate() {
|
||||||
|
if ($this->templateFile !== null) {
|
||||||
|
return parent::getTemplate();
|
||||||
|
} else {
|
||||||
|
return str_replace('CDbMigration', 'ZDbMigration', parent::getTemplate());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
170
protected/components/Controller.php
Normal file
@ -0,0 +1,170 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Controller is the customized base controller class.
|
||||||
|
* All controller classes for this application should extend from this base class.
|
||||||
|
*
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class Controller extends CController {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string the default layout for the controller view. Defaults to '//layouts/column1',
|
||||||
|
* meaning using a single column layout. See 'protected/views/layouts/column1.php'.
|
||||||
|
*/
|
||||||
|
public $layout = '//layouts/main';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string the sub layout for the controller view. Defaults to '',
|
||||||
|
*/
|
||||||
|
public $subLayout = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array context menu items. This property will be assigned to {@link CMenu::items}.
|
||||||
|
*/
|
||||||
|
public $menu = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array the breadcrumbs of the current page. The value of this property will
|
||||||
|
* be assigned to {@link CBreadcrumbs::links}. Please refer to {@link CBreadcrumbs::links}
|
||||||
|
* for more details on how to specify this property.
|
||||||
|
*/
|
||||||
|
public $breadcrumbs = array();
|
||||||
|
public $snippets = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Minimum required database version (HSetting)
|
||||||
|
*
|
||||||
|
* @var type
|
||||||
|
*/
|
||||||
|
#protected $requiredDbVersion = 28;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inits the controller class.
|
||||||
|
*
|
||||||
|
* - Force Installer when not HumHub is not installed yet.
|
||||||
|
* - Registers Event Interceptor
|
||||||
|
* - Loads Basic HSetting
|
||||||
|
* - Set Language
|
||||||
|
* - Check Database Version
|
||||||
|
*/
|
||||||
|
public function init() {
|
||||||
|
|
||||||
|
Yii::app()->interceptor->intercept($this);
|
||||||
|
|
||||||
|
// Force installer, when not installed
|
||||||
|
if (!Yii::app()->params['installed']) {
|
||||||
|
if ($this->getModule() != null && $this->getModule()->id == "installer") {
|
||||||
|
return parent::init();
|
||||||
|
}
|
||||||
|
$this->redirect(array('//installer/index'));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Switch to correct user language
|
||||||
|
if (Yii::app()->user->language) {
|
||||||
|
Yii::app()->language = Yii::app()->user->language;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enable Jquery Globally
|
||||||
|
Yii::app()->clientScript->registerCoreScript('jquery');
|
||||||
|
|
||||||
|
// Tweaks for Ajax Requests
|
||||||
|
Yii::app()->user->loginRequiredAjaxResponse = "<script>window.location.href = '" . Yii::app()->user->id . "';</script>";
|
||||||
|
if (Yii::app()->request->isAjaxRequest) {
|
||||||
|
Yii::app()->clientScript->scriptMap = array(
|
||||||
|
'jquery.js' => false,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set a javascript variable which holds the Name and the Value of the CSRF Variable.
|
||||||
|
Yii::app()->clientScript->setJavascriptVariable('csrfName', Yii::app()->request->csrfTokenName);
|
||||||
|
Yii::app()->clientScript->setJavascriptVariable('csrfValue', Yii::app()->request->csrfToken);
|
||||||
|
Yii::app()->clientScript->setJavascriptVariable('baseUrl', Yii::app()->getBaseUrl(true));
|
||||||
|
|
||||||
|
return parent::init();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create Redirect for AJAX Requests which output goes into HTML content.
|
||||||
|
* Is an alternative method to redirect, for ajax responses.
|
||||||
|
*/
|
||||||
|
public function htmlRedirect($url = "") {
|
||||||
|
|
||||||
|
echo "<script>\n";
|
||||||
|
|
||||||
|
// If current URL == New Url (absolute || relative) then only Refresh
|
||||||
|
echo "if (window.location.pathname+window.location.search+window.location.hash == '" . $url . "' || '" . $url . "' == window.location.href) { \n";
|
||||||
|
|
||||||
|
//echo "window.location.reload();\n"; // Drops warning on Posts
|
||||||
|
// Remove test.php#xy (#xy) part
|
||||||
|
$temp = explode("#", $url);
|
||||||
|
$url = $temp[0];
|
||||||
|
|
||||||
|
echo "if (window.location.search == '') {\n";
|
||||||
|
echo "window.location.href = '" . $url . "?c=" . time() . "';\n";
|
||||||
|
echo "} else { \n";
|
||||||
|
echo "window.location.href = '" . $url . "&c=" . time() . "';\n";
|
||||||
|
echo "} \n";
|
||||||
|
|
||||||
|
// When completly new url, set new window location
|
||||||
|
echo "} else { \n";
|
||||||
|
echo "window.location.href = '" . $url . "';\n";
|
||||||
|
echo "} \n";
|
||||||
|
echo "</script>\n";
|
||||||
|
Yii::app()->end();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Outputs a given JSON Array and ends the application
|
||||||
|
*
|
||||||
|
* @param type $json
|
||||||
|
*/
|
||||||
|
protected function renderJson($json) {
|
||||||
|
echo CJSON::encode($json);
|
||||||
|
Yii::app()->end();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ensures the current request is a post, when not throw an error
|
||||||
|
*/
|
||||||
|
public function forcePostRequest() {
|
||||||
|
if (!Yii::app()->request->isPostRequest) {
|
||||||
|
throw new CHttpException(500, Yii::t('base', 'Invalid request.'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes a modal
|
||||||
|
*/
|
||||||
|
public function renderModalClose() {
|
||||||
|
|
||||||
|
// close modal to hide the loaded view, which is visible for some seconds, after creation
|
||||||
|
echo "<script>";
|
||||||
|
echo "$('#globalModal').modal('hide')";
|
||||||
|
echo "</script>";
|
||||||
|
//Yii::app()->end();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
116
protected/components/HActiveRecord.php
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HActiveRecord is an extended version of the CActiveRecord.
|
||||||
|
*
|
||||||
|
* Each HActiveRecord has some extra meta database fields like:
|
||||||
|
* - created by
|
||||||
|
* - created_at
|
||||||
|
* - updated_by
|
||||||
|
* - updated_at
|
||||||
|
*
|
||||||
|
* The underlying HActiveRecord table must have these fields.
|
||||||
|
*
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
abstract class HActiveRecord extends CActiveRecord {
|
||||||
|
|
||||||
|
// Avoid Errors when fields not exists
|
||||||
|
public $created_at;
|
||||||
|
public $created_by;
|
||||||
|
public $updated_at;
|
||||||
|
public $updated_by;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inits the active records and registers the event interceptor
|
||||||
|
*/
|
||||||
|
public function init() {
|
||||||
|
|
||||||
|
parent::init();
|
||||||
|
|
||||||
|
// Intercept this controller
|
||||||
|
Yii::app()->interceptor->intercept($this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prepares create_time, create_user_id, update_time and update_user_id attributes before performing validation.
|
||||||
|
*/
|
||||||
|
protected function beforeValidate() {
|
||||||
|
|
||||||
|
// Check if we got an user object
|
||||||
|
if (isset(Yii::app()->user)) {
|
||||||
|
$userId = Yii::app()->user->id;
|
||||||
|
} else {
|
||||||
|
$userId = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if ($this->isNewRecord) {
|
||||||
|
// set the create date, last updated date and the user doing the creating
|
||||||
|
$this->created_at = $this->created_at = new CDbExpression('NOW()');
|
||||||
|
if ($this->created_by == "")
|
||||||
|
$this->created_by = $userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dont update in this scenario (UserSpaceMembership, only changed column is "last_visit")
|
||||||
|
if ($this->scenario != 'last_visit' && $userId != 0) {
|
||||||
|
|
||||||
|
//not a new record, so just set the last updated time and last updated user id
|
||||||
|
$this->updated_at = new CDbExpression('NOW()');
|
||||||
|
if ($this->updated_by == "")
|
||||||
|
$this->updated_by = $userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return parent::beforeValidate();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the creator user of this active record
|
||||||
|
* (Faster than relation() because cached).
|
||||||
|
*
|
||||||
|
* @return User
|
||||||
|
*/
|
||||||
|
public function getCreator() {
|
||||||
|
return User::model()->findByPk($this->created_by);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the updater user of this active record
|
||||||
|
* (Faster than relation() because cached).
|
||||||
|
*
|
||||||
|
* @return User
|
||||||
|
*/
|
||||||
|
public function getUpdater() {
|
||||||
|
return User::model()->findByPk($this->updated_by);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a unique id for this record
|
||||||
|
*
|
||||||
|
* @return String Unique Id of this record
|
||||||
|
*/
|
||||||
|
public function getUniqueId() {
|
||||||
|
return get_class($this) . "_" . $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
97
protected/components/HActiveRecordContent.php
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HActiveRecordContent is the base activerecord for content objects.
|
||||||
|
*
|
||||||
|
* Each model which represents a content should derived from it.
|
||||||
|
* (e.g. Post, Question, Task, Note, ...)
|
||||||
|
*
|
||||||
|
* It automatically adds the HContentBehavior which binds a 'contentMeta'
|
||||||
|
* attribute, that always points to a Content Model Record.
|
||||||
|
* (See SiContentBavior for more details.)
|
||||||
|
*
|
||||||
|
* The Content Model is responsible for
|
||||||
|
* - Content to User/Space Binding
|
||||||
|
* - Access Controlling
|
||||||
|
* - Wall Integration
|
||||||
|
* - ...
|
||||||
|
* (See Content Model for more details.)
|
||||||
|
*
|
||||||
|
* Cleanup:
|
||||||
|
* On workspace or user deletion, this objects will automatically deleted.
|
||||||
|
*
|
||||||
|
* Note: Comments, Likes or Files are NOT Content Objects. These objects are
|
||||||
|
* ContentAddons (HActiveRecordContentAddon) which always belongs to one
|
||||||
|
* Content Object.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HActiveRecordContent extends HActiveRecord {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extended Constructor which automatically attaches
|
||||||
|
* the HContentBehavior to the objects.
|
||||||
|
*
|
||||||
|
* @param type $scenario
|
||||||
|
*/
|
||||||
|
public function __construct($scenario = 'insert') {
|
||||||
|
|
||||||
|
parent::__construct($scenario);
|
||||||
|
|
||||||
|
$this->attachBehavior('HContentBehavior', array(
|
||||||
|
'class' => 'application.behaviors.HContentBehavior'
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a short textual title for this content.
|
||||||
|
* Default goes to "Classname (Id)"
|
||||||
|
*
|
||||||
|
* It should be overwritten for a more representative text.
|
||||||
|
*
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function getContentTitle() {
|
||||||
|
$objectModel = get_class($this); // e.g. Post
|
||||||
|
return $objectModel . " (" . $this->getOwner()->id . ")";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the content should also displayed on a wall, overwrite this
|
||||||
|
* method and produce a wall output.
|
||||||
|
*
|
||||||
|
* e.g.
|
||||||
|
* return Yii::app()->getController()->widget('application.modules.myModule.MyContentWidget',
|
||||||
|
* array('myContent' => $this),
|
||||||
|
* true
|
||||||
|
* );
|
||||||
|
*
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function getWallOut() {
|
||||||
|
return "Default Wall Output for Class " . get_class($this->getOwner());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
34
protected/components/HActiveRecordContentAddon.php
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HActiveRecordContentAddon is the base active record for content addons.
|
||||||
|
*
|
||||||
|
* Content addons are content types as example Likes, Comments or Files.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HActiveRecordContentAddon extends HActiveRecord {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
69
protected/components/HAjaxLinkPager.php
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HAjaxLinkPager extends the CLinkPager with some ajax functions.
|
||||||
|
*
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HAjaxLinkPager extends CLinkPager {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var String html dom id of target
|
||||||
|
*/
|
||||||
|
public $ajaxContentTarget = "SomeCoolDivId";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var ind counter
|
||||||
|
*/
|
||||||
|
protected $ajaxIdCounter = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a page button.
|
||||||
|
*
|
||||||
|
* You may override this method to customize the page buttons.
|
||||||
|
* @param string $label the text label for the button
|
||||||
|
* @param integer $page the page number
|
||||||
|
* @param string $class the CSS class for the page button. This could be 'page', 'first', 'last', 'next' or 'previous'.
|
||||||
|
* @param boolean $hidden whether this page button is visible
|
||||||
|
* @param boolean $selected whether this page button is selected
|
||||||
|
*
|
||||||
|
* @return string the generated button
|
||||||
|
*/
|
||||||
|
protected function createPageButton($label, $page, $class, $hidden, $selected) {
|
||||||
|
|
||||||
|
$this->ajaxIdCounter++; // For Unique Ajax Link IDs
|
||||||
|
|
||||||
|
if ($hidden || $selected)
|
||||||
|
$class.=' ' . ($hidden ? self::CSS_HIDDEN_PAGE : self::CSS_SELECTED_PAGE);
|
||||||
|
|
||||||
|
$ajaxLink = HHtml::ajaxLink(
|
||||||
|
$label, $this->createPageUrl($page), array(
|
||||||
|
'success' => "function(html) { jQuery('" . $this->ajaxContentTarget . "').replaceWith(html); }",
|
||||||
|
), array(
|
||||||
|
'id' => "siPaginatorLink_" . $page . "_" . $this->ajaxIdCounter,
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return '<li class="' . $class . '">' . $ajaxLink . '</li>';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
129
protected/components/HClientScript.php
Normal file
@ -0,0 +1,129 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HClientScript extends the CClientScript
|
||||||
|
*
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HClientScript extends CClientScript {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array the registered HTML code blocks (key => code)
|
||||||
|
*/
|
||||||
|
public $htmls = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method to inject a javascript variable
|
||||||
|
*
|
||||||
|
* @param String $name
|
||||||
|
* @param String $value
|
||||||
|
*/
|
||||||
|
public function setJavascriptVariable($name, $value) {
|
||||||
|
|
||||||
|
$jsCode = "var " . $name . " = '" . $value . "';\n";
|
||||||
|
$this->registerScript('jsVar_' . $name, $jsCode, CClientScript::POS_BEGIN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a script file
|
||||||
|
*
|
||||||
|
* @param String $src
|
||||||
|
* @param String $position
|
||||||
|
*/
|
||||||
|
public function registerScriptFile($src, $position = NULL) {
|
||||||
|
|
||||||
|
# if(Yii::app()->getRequest()->getIsAjaxRequest()) {
|
||||||
|
# Yii::app()->clientScript->registerScript('loadScript_'.$src, "loadJavaScript(\"".$src."\");", CClientScript::POS_BEGIN);
|
||||||
|
# } else {
|
||||||
|
return parent::registerScriptFile($src, $position);
|
||||||
|
# }
|
||||||
|
#registerScriptFile
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a piece of html code.
|
||||||
|
* @param string $id ID that uniquely identifies this piece of HTML code
|
||||||
|
* @param string $html the html code
|
||||||
|
*
|
||||||
|
* @return CClientScript the CClientScript object itself (to support method chaining, available since version 1.1.5).
|
||||||
|
*/
|
||||||
|
public function registerHtml($id, $html) {
|
||||||
|
|
||||||
|
$this->hasScripts = true;
|
||||||
|
|
||||||
|
$this->htmls[$id] = $html;
|
||||||
|
|
||||||
|
$params = func_get_args();
|
||||||
|
$this->recordCachingAction('clientScript', 'registerScript', $params);
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inserts the scripts and other html snippets at the end of the body section.
|
||||||
|
*
|
||||||
|
* @param string $output the output to be inserted with scripts.
|
||||||
|
*/
|
||||||
|
public function renderBodyEnd(&$output) {
|
||||||
|
if (!isset($this->scriptFiles[self::POS_END]) && !isset($this->scripts[self::POS_END]) && !isset($this->scripts[self::POS_READY]) && !isset($this->scripts[self::POS_LOAD]) && !count($this->htmls) == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
$fullPage = 0;
|
||||||
|
$output = preg_replace('/(<\\/body\s*>)/is', '<###end###>$1', $output, 1, $fullPage);
|
||||||
|
$html = '';
|
||||||
|
if (isset($this->scriptFiles[self::POS_END])) {
|
||||||
|
foreach ($this->scriptFiles[self::POS_END] as $scriptFile)
|
||||||
|
$html.=CHtml::scriptFile($scriptFile) . "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------------------------------
|
||||||
|
// Begin Custom
|
||||||
|
foreach ($this->htmls as $id => $htmlSnippet) {
|
||||||
|
$html .= $htmlSnippet;
|
||||||
|
}
|
||||||
|
//----------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
$scripts = isset($this->scripts[self::POS_END]) ? $this->scripts[self::POS_END] : array();
|
||||||
|
if (isset($this->scripts[self::POS_READY])) {
|
||||||
|
if ($fullPage)
|
||||||
|
$scripts[] = "jQuery(function($) {\n" . implode("\n", $this->scripts[self::POS_READY]) . "\n});";
|
||||||
|
else
|
||||||
|
$scripts[] = implode("\n", $this->scripts[self::POS_READY]);
|
||||||
|
}
|
||||||
|
if (isset($this->scripts[self::POS_LOAD])) {
|
||||||
|
if ($fullPage)
|
||||||
|
$scripts[] = "jQuery(window).on('load',function() {\n" . implode("\n", $this->scripts[self::POS_LOAD]) . "\n});";
|
||||||
|
else
|
||||||
|
$scripts[] = implode("\n", $this->scripts[self::POS_LOAD]);
|
||||||
|
}
|
||||||
|
if (!empty($scripts))
|
||||||
|
$html.=CHtml::script(implode("\n", $scripts)) . "\n";
|
||||||
|
|
||||||
|
if ($fullPage)
|
||||||
|
$output = str_replace('<###end###>', $html, $output);
|
||||||
|
else
|
||||||
|
$output = $output . $html;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
76
protected/components/HConsoleApplication.php
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HConsoleApplication is used as base console application.
|
||||||
|
*
|
||||||
|
* HConsoleApplication extends the default console application with some
|
||||||
|
* functionalities about events and theming support.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.8
|
||||||
|
*/
|
||||||
|
class HConsoleApplication extends CConsoleApplication {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Current theme name
|
||||||
|
*
|
||||||
|
* @var String
|
||||||
|
*/
|
||||||
|
public $theme;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes the console application and setup some event handlers
|
||||||
|
*/
|
||||||
|
protected function init() {
|
||||||
|
|
||||||
|
parent::init();
|
||||||
|
|
||||||
|
$this->interceptor->start();
|
||||||
|
$this->interceptor->intercept($this);
|
||||||
|
|
||||||
|
$this->moduleManager->start();
|
||||||
|
|
||||||
|
if ($this->hasEventHandler('onInit'))
|
||||||
|
$this->onInit(new CEvent($this));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Raised after the application inits.
|
||||||
|
* @param CEvent $event the event parameter
|
||||||
|
*/
|
||||||
|
public function onInit($event) {
|
||||||
|
$this->raiseEvent('onInit', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds a new command to the Console Application
|
||||||
|
*
|
||||||
|
* @param String $name
|
||||||
|
* @param String $file
|
||||||
|
*/
|
||||||
|
public function addCommand($name, $file) {
|
||||||
|
$this->getCommandRunner()->commands[$name] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
50
protected/components/HConsoleApplicationBehavior.php
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HConsoleApplicationBehavior is added to all console applications.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HConsoleApplicationBehavior extends CBehavior {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds a null view renderer.
|
||||||
|
*
|
||||||
|
* @return null
|
||||||
|
*/
|
||||||
|
public function getViewRenderer() {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns null theme.
|
||||||
|
*
|
||||||
|
* @return null
|
||||||
|
*/
|
||||||
|
public function getTheme() {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
73
protected/components/HConsoleCommand.php
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HConsoleCommand extends the default CConsoleCommand.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HConsoleCommand extends CConsoleCommand {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inits the command and prepares the base environment.
|
||||||
|
*/
|
||||||
|
public function init() {
|
||||||
|
|
||||||
|
Yii::app()->interceptor->intercept($this);
|
||||||
|
|
||||||
|
Yii::import('application.vendors.*');
|
||||||
|
require_once('adLDAP/adLDAP.php');
|
||||||
|
|
||||||
|
EZendAutoloader::$prefixes = array('Zend', 'Custom');
|
||||||
|
Yii::import("ext.yiiext.components.zendAutoloader.EZendAutoloader", true);
|
||||||
|
Yii::registerAutoloader(array("EZendAutoloader", "loadClass"), true);
|
||||||
|
|
||||||
|
ini_set('max_execution_time', 9000);
|
||||||
|
ini_set('memory_limit', '1024M');
|
||||||
|
date_default_timezone_set("Europe/Berlin");
|
||||||
|
|
||||||
|
error_reporting(E_ALL ^ E_NOTICE);
|
||||||
|
|
||||||
|
#HSetting::InstallBase();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prints out the console application header
|
||||||
|
*
|
||||||
|
* @param String $title is the title of the command
|
||||||
|
*/
|
||||||
|
protected function printHeader($title) {
|
||||||
|
|
||||||
|
print "
|
||||||
|
|
||||||
|
|
||||||
|
HumHub Console Interface - Version 1
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
$title
|
||||||
|
--------------------------------------------------------------------------------\n
|
||||||
|
\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
49
protected/components/HHttpRequest.php
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HHttpRequest extends the CHttpRequest.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HHttpRequest extends CHttpRequest {
|
||||||
|
|
||||||
|
public $csrfTokenName = 'CSRF_TOKEN';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns whether this is an AJAX (XMLHttpRequest) request.
|
||||||
|
* @return boolean whether this is an AJAX (XMLHttpRequest) request.
|
||||||
|
*/
|
||||||
|
public function getIsAjaxRequest() {
|
||||||
|
|
||||||
|
if (!parent::getIsAjaxRequest()) {
|
||||||
|
if (isset($_REQUEST['ajax'])) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
92
protected/components/HInterceptor.php
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HInterceptor allows modules to attach events and behaviors to all SIComponents.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HInterceptor extends CApplicationComponent {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Array list of events and callbacks
|
||||||
|
*/
|
||||||
|
private $events = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts interceptor service
|
||||||
|
*/
|
||||||
|
public function start() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Intercepts a given object instance
|
||||||
|
*
|
||||||
|
* @param Object $obj
|
||||||
|
*/
|
||||||
|
public function intercept($obj) {
|
||||||
|
|
||||||
|
// Attach Event Handler for this class
|
||||||
|
$className = get_class($obj);
|
||||||
|
if (isset($this->events[$className])) {
|
||||||
|
foreach ($this->events[$className] as $event) {
|
||||||
|
$obj->attachEventHandler($event[0], $event[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Install also for parent classes
|
||||||
|
$parentClassName = get_parent_class($className);
|
||||||
|
while ($parentClassName !== false) {
|
||||||
|
if (isset($this->events[$parentClassName])) {
|
||||||
|
foreach ($this->events[$parentClassName] as $event) {
|
||||||
|
$obj->attachEventHandler($event[0], $event[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$parentClassName = get_parent_class($parentClassName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attaches an event handler
|
||||||
|
*
|
||||||
|
* @param String $className
|
||||||
|
* @param String $eventName
|
||||||
|
* @param type $handler
|
||||||
|
*/
|
||||||
|
public function attachEventHandler($className, $eventName, $handler=null) {
|
||||||
|
|
||||||
|
# if ($className == 'ModuleManager') {#
|
||||||
|
# // There is already an instance
|
||||||
|
# Yii::app()->moduleManager->attachEventHandler($eventName, $handler);
|
||||||
|
# return;
|
||||||
|
#}
|
||||||
|
|
||||||
|
if (!isset($this->events[$className]))
|
||||||
|
$this->events[$className] = array();
|
||||||
|
|
||||||
|
$this->events[$className][] = array($eventName, $handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
97
protected/components/HMailMessage.php
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HMailMessage is a changed version of the YiiMailMessage class.
|
||||||
|
* Its supports better theming support and allows mail views in different folders.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HMailMessage extends YiiMailMessage {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the body of this entity, either as a string, or array of view
|
||||||
|
* variables if a view is set, or as an instance of
|
||||||
|
* {@link Swift_OutputByteStream}.
|
||||||
|
*
|
||||||
|
* @param mixed the body of the message. If a $this->view is set and this
|
||||||
|
* is a string, this is passed to the view as $body. If $this->view is set
|
||||||
|
* and this is an array, the array values are passed to the view like in the
|
||||||
|
* controller render() method
|
||||||
|
* @param string content type optional. For html, set to 'html/text'
|
||||||
|
* @param string charset optional
|
||||||
|
*/
|
||||||
|
public function setBody($body = '', $contentType = null, $charset = null) {
|
||||||
|
if ($this->view !== null) {
|
||||||
|
if (!is_array($body))
|
||||||
|
$body = array('body' => $body);
|
||||||
|
|
||||||
|
// if Yii::app()->controller doesn't exist create a dummy
|
||||||
|
// controller to render the view (needed in the console app)
|
||||||
|
if (isset(Yii::app()->controller))
|
||||||
|
$controller = Yii::app()->controller;
|
||||||
|
else
|
||||||
|
$controller = new CController('YiiMail');
|
||||||
|
|
||||||
|
// File Path to Template
|
||||||
|
$viewPath = "";
|
||||||
|
|
||||||
|
// Name of current theme if enabled
|
||||||
|
$themeName = "";
|
||||||
|
|
||||||
|
if (Yii::app() instanceOf CConsoleApplication) {
|
||||||
|
// ConsoleApplication
|
||||||
|
Yii::setPathOfAlias('webroot', '../' . dirname($_SERVER['SCRIPT_FILENAME']));
|
||||||
|
|
||||||
|
if (Yii::app()->theme && Yii::app()->theme != "") {
|
||||||
|
$themeName = Yii::app()->theme;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// WebApplication
|
||||||
|
if (($theme = Yii::app()->getTheme()) !== null) {
|
||||||
|
$themeName = $theme->getName();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When ThemeName is speified
|
||||||
|
if ($themeName != "") {
|
||||||
|
$viewThemed = str_replace('application.views.', 'webroot.themes.' . $themeName . '.views.', $this->view);
|
||||||
|
$viewThemed = preg_replace('/application\.modules(?:_core)?\.(.*?)\.views\.(.*)/i', 'webroot.themes.' . $themeName . '.views.\1.\2', $viewThemed);
|
||||||
|
|
||||||
|
if (file_exists(Yii::getPathOfAlias($viewThemed) . ".php")) {
|
||||||
|
$viewPath = Yii::getPathOfAlias($viewThemed) . ".php";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use orginal view name, if not set yet
|
||||||
|
if ($viewPath == "") {
|
||||||
|
$viewPath = Yii::getPathOfAlias($this->view) . ".php";
|
||||||
|
}
|
||||||
|
|
||||||
|
$body = $controller->renderInternal($viewPath, array_merge($body, array('mail' => $this)), true);
|
||||||
|
}
|
||||||
|
return $this->message->setBody($body, $contentType, $charset);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
140
protected/components/HPhpMessageSource.php
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HPhpMessageSource is a changed version of CPhpMessageSource class.
|
||||||
|
*
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HPhpMessageSource extends CPhpMessageSource {
|
||||||
|
|
||||||
|
const CACHE_KEY_PREFIX = 'Yii.CPhpMessageSource.';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var integer the time in seconds that the messages can remain valid in cache.
|
||||||
|
* Defaults to 0, meaning the caching is disabled.
|
||||||
|
*/
|
||||||
|
public $cachingDuration = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string the ID of the cache application component that is used to cache the messages.
|
||||||
|
* Defaults to 'cache' which refers to the primary cache application component.
|
||||||
|
* Set this property to false if you want to disable caching the messages.
|
||||||
|
*/
|
||||||
|
public $cacheID = 'cache';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string the base path for all translated messages. Defaults to null, meaning
|
||||||
|
* the "messages" subdirectory of the application directory (e.g. "protected/messages").
|
||||||
|
*/
|
||||||
|
public $basePath;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array the message paths for extensions that do not have a base class to use as category prefix.
|
||||||
|
* The format of the array should be:
|
||||||
|
* <pre>
|
||||||
|
* array(
|
||||||
|
* 'ExtensionName' => 'ext.ExtensionName.messages',
|
||||||
|
* )
|
||||||
|
* </pre>
|
||||||
|
* Where the key is the name of the extension and the value is the alias to the path
|
||||||
|
* of the "messages" subdirectory of the extension.
|
||||||
|
* When using Yii::t() to translate an extension message, the category name should be
|
||||||
|
* set as 'ExtensionName.categoryName'.
|
||||||
|
* Defaults to an empty array, meaning no extensions registered.
|
||||||
|
* @since 1.1.13
|
||||||
|
*/
|
||||||
|
public $extensionPaths = array();
|
||||||
|
private $_files = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes the application component.
|
||||||
|
* This method overrides the parent implementation by preprocessing
|
||||||
|
* the user request data.
|
||||||
|
*/
|
||||||
|
public function init() {
|
||||||
|
parent::init();
|
||||||
|
if ($this->basePath === null)
|
||||||
|
$this->basePath = Yii::getPathOfAlias('application.messages');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines the message file name based on the given category and language.
|
||||||
|
* If the category name contains a dot, it will be split into the module class name and the category name.
|
||||||
|
* In this case, the message file will be assumed to be located within the 'messages' subdirectory of
|
||||||
|
* the directory containing the module class file.
|
||||||
|
* Otherwise, the message file is assumed to be under the {@link basePath}.
|
||||||
|
* @param string $category category name
|
||||||
|
* @param string $language language ID
|
||||||
|
* @return string the message file path
|
||||||
|
*/
|
||||||
|
protected function getMessageFile($category, $language) {
|
||||||
|
if (!isset($this->_files[$category][$language])) {
|
||||||
|
if (($pos = strpos($category, '.')) !== false) {
|
||||||
|
$extensionClass = substr($category, 0, $pos);
|
||||||
|
$extensionCategory = substr($category, $pos + 1);
|
||||||
|
// First check if there's an extension registered for this class.
|
||||||
|
if (isset($this->extensionPaths[$extensionClass]))
|
||||||
|
$this->_files[$category][$language] = Yii::getPathOfAlias($this->extensionPaths[$extensionClass]) . DIRECTORY_SEPARATOR . $language . DIRECTORY_SEPARATOR . $extensionCategory . '.php';
|
||||||
|
else {
|
||||||
|
// No extension registered, need to find it.
|
||||||
|
try {
|
||||||
|
$class = new ReflectionClass($extensionClass);
|
||||||
|
$this->_files[$category][$language] = dirname($class->getFileName()) . DIRECTORY_SEPARATOR . 'messages' . DIRECTORY_SEPARATOR . $language . DIRECTORY_SEPARATOR . $extensionCategory . '.php';
|
||||||
|
} catch (Exception $e) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
$this->_files[$category][$language] = $this->basePath . DIRECTORY_SEPARATOR . $language . DIRECTORY_SEPARATOR . $category . '.php';
|
||||||
|
}
|
||||||
|
return $this->_files[$category][$language];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the message translation for the specified language and category.
|
||||||
|
* @param string $category the message category
|
||||||
|
* @param string $language the target language
|
||||||
|
* @return array the loaded messages
|
||||||
|
*/
|
||||||
|
protected function loadMessages($category, $language) {
|
||||||
|
$messageFile = $this->getMessageFile($category, $language);
|
||||||
|
|
||||||
|
if ($this->cachingDuration > 0 && $this->cacheID !== false && ($cache = Yii::app()->getComponent($this->cacheID)) !== null) {
|
||||||
|
$key = self::CACHE_KEY_PREFIX . $messageFile;
|
||||||
|
if (($data = $cache->get($key)) !== false)
|
||||||
|
return unserialize($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_file($messageFile)) {
|
||||||
|
$messages = include($messageFile);
|
||||||
|
if (!is_array($messages))
|
||||||
|
$messages = array();
|
||||||
|
if (isset($cache)) {
|
||||||
|
$dependency = new CFileCacheDependency($messageFile);
|
||||||
|
$cache->set($key, serialize($messages), $this->cachingDuration, $dependency);
|
||||||
|
}
|
||||||
|
return $messages;
|
||||||
|
} else
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
69
protected/components/HTheme.php
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HTheme is an overwrite of CTheme
|
||||||
|
*
|
||||||
|
* This is caused by our view path for modules is also separeted into modules/ folders.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HTheme extends CTheme {
|
||||||
|
/**
|
||||||
|
* Finds the view file for the specified controller's view.
|
||||||
|
*
|
||||||
|
* @param CController $controller the controller
|
||||||
|
* @param string $viewName the view name
|
||||||
|
* @return string the view file path. False if the file does not exist.
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
public function getViewFile($controller, $viewName) {
|
||||||
|
$moduleViewPath = $this->getViewPath();
|
||||||
|
if (($module = $controller->getModule()) !== null) {
|
||||||
|
$moduleViewPath.='/modules/' . $module->getId();
|
||||||
|
# ^^^^^^ added modules here
|
||||||
|
}
|
||||||
|
return $controller->resolveViewFile($viewName, $this->getViewPath() . '/' . $controller->getUniqueId(), $this->getViewPath(), $moduleViewPath);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an array of all installed themes.
|
||||||
|
*
|
||||||
|
* @return Array
|
||||||
|
*/
|
||||||
|
public static function getThemes() {
|
||||||
|
$themes = array();
|
||||||
|
$themePath = Yii::getPathOfAlias('webroot') . DIRECTORY_SEPARATOR . "themes";
|
||||||
|
|
||||||
|
foreach (scandir($themePath) as $file) {
|
||||||
|
if ($file == "." || $file == ".." || !is_dir($themePath . DIRECTORY_SEPARATOR . $file)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$themes[$file] = $file;
|
||||||
|
}
|
||||||
|
return $themes;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
150
protected/components/HWidget.php
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HWidget is the base class for all widgets.
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class HWidget extends CWidget {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array view paths for different types of widgets
|
||||||
|
*/
|
||||||
|
private static $_viewPaths;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the view path for this widget.
|
||||||
|
*
|
||||||
|
* Transform:
|
||||||
|
* /protected/modules/moduleId/widgets/views/ to /themes/mytheme/views/moduleId/widgets
|
||||||
|
* /protected/widgets/views/ to /themes/mytheme/views/widgets
|
||||||
|
*
|
||||||
|
* Instead of normal yii behavior:
|
||||||
|
* /themes/myTheme/WidgetClass/..
|
||||||
|
*
|
||||||
|
* @param type $checkTheme
|
||||||
|
* @return String
|
||||||
|
*/
|
||||||
|
public function getViewPath($checkTheme = false) {
|
||||||
|
|
||||||
|
// Fastlane
|
||||||
|
$className = get_class($this);
|
||||||
|
if (isset(self::$_viewPaths[$className]))
|
||||||
|
return self::$_viewPaths[$className];
|
||||||
|
|
||||||
|
if ($checkTheme && ($theme = Yii::app()->getTheme()) !== null) {
|
||||||
|
// /themes/myTheme/views/
|
||||||
|
$path = $theme->getViewPath() . DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
// + moduleId/ - if exists
|
||||||
|
$moduleId = $this->getModuleId();
|
||||||
|
if ($moduleId)
|
||||||
|
$path .= $moduleId . DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
$path .= 'widgets' . DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
if (is_dir($path))
|
||||||
|
return self::$_viewPaths[$className] = $path;
|
||||||
|
}
|
||||||
|
|
||||||
|
$class = new ReflectionClass($className);
|
||||||
|
return self::$_viewPaths[$className] = dirname($class->getFileName()) . DIRECTORY_SEPARATOR . 'views';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extends CWidgets getViewFile by possibilty to get also themed version
|
||||||
|
* of a dotted view Filename
|
||||||
|
*
|
||||||
|
* @param string $viewName name of the view (without file extension)
|
||||||
|
* @return string the view file path. False if the view file does not exist
|
||||||
|
* @see CApplication::findLocalizedFile
|
||||||
|
*/
|
||||||
|
public function getViewFile($viewName) {
|
||||||
|
|
||||||
|
// a path alias e.g. application.modules.x.y.z.
|
||||||
|
if (strpos($viewName, '.')) {
|
||||||
|
|
||||||
|
|
||||||
|
if (($theme = Yii::app()->getTheme()) !== null) {
|
||||||
|
// Replace application.modules[_core].MODULEID.widgets.views
|
||||||
|
// in
|
||||||
|
// webroot.themes.CURRENTTHEME.views.MODULEID.widgets
|
||||||
|
$viewNameTheme = $viewName;
|
||||||
|
$viewNameTheme = str_replace('application.views.', 'webroot.themes.' . $theme->getName() . '.views.', $viewNameTheme);
|
||||||
|
$viewNameTheme = preg_replace('/application\.modules(?:_core)?\.(.*?)\.views\.(.*)/i', 'webroot.themes.' . $theme->getName() . '.views.\1.\2', $viewNameTheme);
|
||||||
|
#$viewNameTheme = preg_replace('/widgets\.views/', 'widgets', $viewNameTheme);
|
||||||
|
|
||||||
|
$viewFile = Yii::getPathOfAlias($viewNameTheme);
|
||||||
|
|
||||||
|
// Check if File exists
|
||||||
|
if (is_file($viewFile . '.php')) {
|
||||||
|
return Yii::app()->findLocalizedFile($viewFile . '.php');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Fall Back to default
|
||||||
|
}
|
||||||
|
return parent::getViewFile($viewName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the parent moduleId if can determined.
|
||||||
|
* This requires that the module is located under /modules/ or /modules_core/.
|
||||||
|
*
|
||||||
|
* It doesn´t support nested modules.
|
||||||
|
* Always the parent moduleId is returned.
|
||||||
|
*
|
||||||
|
* @return String The Id of Module
|
||||||
|
*/
|
||||||
|
public function getModuleId() {
|
||||||
|
|
||||||
|
// Get Directory of current widget class
|
||||||
|
$reflector = new ReflectionClass(get_class($this));
|
||||||
|
$fileName = $reflector->getFileName();
|
||||||
|
|
||||||
|
// Search for .../modules_core/FINDME/... or .../modules/FINDME/...
|
||||||
|
preg_match('/\/modules(?:_core)?\/(.*?)\//', $fileName, $match);
|
||||||
|
|
||||||
|
if (isset($match[1]))
|
||||||
|
return $match[1];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds the createUrl functionality to widget by passing it to the current
|
||||||
|
* controller.
|
||||||
|
*
|
||||||
|
* @param String $action
|
||||||
|
* @param Array $params
|
||||||
|
* @param String $ampersand
|
||||||
|
* @return String url
|
||||||
|
*/
|
||||||
|
public function createUrl($action, $params = array(), $ampersand = '&') {
|
||||||
|
return $this->getController()->createUrl($this->actionPrefix . $action, $params, $ampersand);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
411
protected/components/ModuleManager.php
Normal file
@ -0,0 +1,411 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ModuleManager allows dynamic enabling/disabling of application modules.
|
||||||
|
*
|
||||||
|
* Each module has a autostart.php which can register the module.
|
||||||
|
*
|
||||||
|
* Modules must register with a module definition, which holds all relevant
|
||||||
|
* information about it.
|
||||||
|
*
|
||||||
|
* Module Definition Array:
|
||||||
|
* id => mymodule (also folder name under /modules/...)
|
||||||
|
* title => My Module
|
||||||
|
* icon => cssClass
|
||||||
|
* description => someText (For Admin Manage Modules)
|
||||||
|
* isSpaceModule => true/FALSE (Is a workspace module)
|
||||||
|
* isCoreModule => true/FALSE (Is core module, always enabled)
|
||||||
|
* configRoute => 'mymodule/configure' (Configuration URL for SuperAdmin)
|
||||||
|
*
|
||||||
|
* @todo cache enabled modules - problem module manager started before caching
|
||||||
|
*
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class ModuleManager extends CApplicationComponent {
|
||||||
|
|
||||||
|
const AUTOSTART_CACHE_FILE_NAME = "cache_autostart.php";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Array of all registered module definitions
|
||||||
|
*/
|
||||||
|
public $registeredModules;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Array of enabled module ids.
|
||||||
|
*/
|
||||||
|
public $enabledModules;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Array of registered content model classes.
|
||||||
|
*/
|
||||||
|
public $registeredContentModels = array();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes the application component.
|
||||||
|
* This should also should check which module is enabled
|
||||||
|
*/
|
||||||
|
public function init() {
|
||||||
|
|
||||||
|
parent::init();
|
||||||
|
|
||||||
|
if (Yii::app()->params['installed'])
|
||||||
|
$this->loadEnabledModules();
|
||||||
|
|
||||||
|
// Intercept this controller
|
||||||
|
Yii::app()->interceptor->intercept($this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function start() {
|
||||||
|
|
||||||
|
$this->executeAutoloaders();
|
||||||
|
#print "start";
|
||||||
|
#die();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Searches and executes all module autoloaders.
|
||||||
|
*
|
||||||
|
* The module autoloaders are stored in a file "autostart.php" which can be
|
||||||
|
* placed in the root directory of the module.
|
||||||
|
*
|
||||||
|
* @todo Caching autostarts
|
||||||
|
* @todo Remove rendundant code
|
||||||
|
*/
|
||||||
|
private function executeAutoloaders() {
|
||||||
|
|
||||||
|
$cacheFileName = Yii::app()->getRuntimePath() . DIRECTORY_SEPARATOR . self::AUTOSTART_CACHE_FILE_NAME;
|
||||||
|
if (file_exists($cacheFileName)) {
|
||||||
|
require_once($cacheFileName);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$fileNames = array();
|
||||||
|
|
||||||
|
// Looking up 3rd party modules
|
||||||
|
$modulesPaths = array(Yii::app()->getBasePath() . DIRECTORY_SEPARATOR . 'modules', Yii::app()->getBasePath() . DIRECTORY_SEPARATOR . 'modules_core');
|
||||||
|
|
||||||
|
// Execute Autoloaders in each modules paths
|
||||||
|
foreach ($modulesPaths as $modulesPath) {
|
||||||
|
|
||||||
|
// Scan Modules
|
||||||
|
$modules = scandir($modulesPath);
|
||||||
|
foreach ($modules as $module) {
|
||||||
|
if ($module == '.' || $module == '..')
|
||||||
|
continue;
|
||||||
|
|
||||||
|
$moduleDir = $modulesPath . DIRECTORY_SEPARATOR . $module . DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
if (is_dir($moduleDir) && is_file($moduleDir . 'autostart.php')) {
|
||||||
|
|
||||||
|
// Store Filename to Cache Content
|
||||||
|
$fileNames[] = $moduleDir . 'autostart.php';
|
||||||
|
|
||||||
|
// Execute Autoloader
|
||||||
|
require_once($moduleDir . 'autostart.php');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Created a file which contains all autoloaders
|
||||||
|
$content = "";
|
||||||
|
foreach ($fileNames as $fileName) {
|
||||||
|
$content .= file_get_contents($fileName);
|
||||||
|
}
|
||||||
|
file_put_contents($cacheFileName, $content);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads all enabled modules from the database. (Cached)
|
||||||
|
*/
|
||||||
|
private function loadEnabledModules() {
|
||||||
|
|
||||||
|
$cacheId = "enabledModules";
|
||||||
|
$cacheValue = Yii::app()->cache->get($cacheId);
|
||||||
|
|
||||||
|
if ($cacheValue === false || !is_array($cacheValue)) {
|
||||||
|
|
||||||
|
$enabledModules = array();
|
||||||
|
foreach (ModuleEnabled::model()->findAll() as $em) {
|
||||||
|
$enabledModules[$em->module_id] = $em->module_id;
|
||||||
|
}
|
||||||
|
Yii::app()->cache->set($cacheId, $enabledModules, HSetting::Get('expireTime', 'cache'));
|
||||||
|
$this->enabledModules = $enabledModules;
|
||||||
|
} else {
|
||||||
|
$this->enabledModules = $cacheValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flushes Module Managers cache
|
||||||
|
*/
|
||||||
|
public function flushCache() {
|
||||||
|
|
||||||
|
// Autoloader Cache File
|
||||||
|
$cacheFileName = Yii::app()->getRuntimePath() . DIRECTORY_SEPARATOR . self::AUTOSTART_CACHE_FILE_NAME;
|
||||||
|
if (file_exists($cacheFileName)) {
|
||||||
|
unlink($cacheFileName);
|
||||||
|
}
|
||||||
|
|
||||||
|
$cacheId = "enabledModules";
|
||||||
|
Yii::app()->cache->delete($cacheId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a module
|
||||||
|
* This is usally called in the autostart file of the module.
|
||||||
|
*
|
||||||
|
* @param Array $definition
|
||||||
|
*/
|
||||||
|
public function register($definition) {
|
||||||
|
$id = $definition['id'];
|
||||||
|
|
||||||
|
if (!isset($definition['isSpaceModule']))
|
||||||
|
$definition['isSpaceModule'] = false;
|
||||||
|
|
||||||
|
if (!isset($definition['isCoreModule']))
|
||||||
|
$definition['isCoreModule'] = false;
|
||||||
|
|
||||||
|
if (!isset($definition['configRoute']))
|
||||||
|
$definition['configRoute'] = '';
|
||||||
|
|
||||||
|
if (!isset($definition['spaceConfigRoute']))
|
||||||
|
$definition['spaceConfigRoute'] = '';
|
||||||
|
|
||||||
|
|
||||||
|
$this->registeredModules[$id] = $definition;
|
||||||
|
|
||||||
|
// Check if module is enabled
|
||||||
|
if (Yii::app()->moduleManager->isEnabled($id)) {
|
||||||
|
|
||||||
|
// Register Yii Module
|
||||||
|
if (isset($definition['class'])) {
|
||||||
|
Yii::app()->setModules(array(
|
||||||
|
$id => array(
|
||||||
|
'class' => $definition['class']
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set Imports
|
||||||
|
if (isset($definition['import'])) {
|
||||||
|
Yii::app()->setImport($definition['import']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Register Event Handlers
|
||||||
|
if (isset($definition['events'])) {
|
||||||
|
foreach ($definition['events'] as $event) {
|
||||||
|
Yii::app()->interceptor->attachEventHandler(
|
||||||
|
$event['class'], $event['event'], $event['callback']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if a module is enabled or not.
|
||||||
|
*
|
||||||
|
* @param type $moduleId
|
||||||
|
* @return boolean
|
||||||
|
*/
|
||||||
|
public function isEnabled($moduleId) {
|
||||||
|
|
||||||
|
$definition = $this->getDefinition($moduleId);
|
||||||
|
|
||||||
|
if ($definition['isCoreModule'])
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// Core installed yet?
|
||||||
|
if (!Yii::app()->params['installed'])
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (in_array($moduleId, $this->enabledModules)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#$moduleEnabled = ModuleEnabled::model()->findByPk($moduleId);
|
||||||
|
#if ($moduleEnabled != null) {
|
||||||
|
# return true;
|
||||||
|
#}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an array with all registered modules
|
||||||
|
* This contains all enabled & disabled modules.
|
||||||
|
* Key is the moduleId and value is the module definition
|
||||||
|
*
|
||||||
|
* @return type
|
||||||
|
*/
|
||||||
|
public function getRegisteredModules() {
|
||||||
|
return $this->registeredModules;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an array with enabled modules
|
||||||
|
* Key of the array is the module id and value is the module definition.
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function getEnabledModules() {
|
||||||
|
|
||||||
|
$enabledModules = array();
|
||||||
|
|
||||||
|
foreach ($this->getRegisteredModules() as $moduleId => $definition) {
|
||||||
|
if ($this->isEnabled($moduleId)) {
|
||||||
|
$enabledModules[$moduleId] = $definition;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $enabledModules;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enables a module by given module id.
|
||||||
|
*
|
||||||
|
* @param String $id
|
||||||
|
*/
|
||||||
|
public function enable($id) {
|
||||||
|
|
||||||
|
$definition = $this->getDefinition($id);
|
||||||
|
if ($definition != null) {
|
||||||
|
|
||||||
|
// Core Modules doesn´t need to enabled
|
||||||
|
if (!$definition['isCoreModule']) {
|
||||||
|
|
||||||
|
$moduleEnabled = ModuleEnabled::model()->findByPk($id);
|
||||||
|
if ($moduleEnabled == null) {
|
||||||
|
$moduleEnabled = new ModuleEnabled();
|
||||||
|
$moduleEnabled->module_id = $id;
|
||||||
|
$moduleEnabled->save();
|
||||||
|
|
||||||
|
// Fire Event Disabled Event
|
||||||
|
if ($this->hasEventHandler('onEnable'))
|
||||||
|
$this->onEnable(new CEvent($this, $id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->flushCache();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Disables a active module by given module id
|
||||||
|
*
|
||||||
|
* @param String $id
|
||||||
|
*/
|
||||||
|
public function disable($id) {
|
||||||
|
|
||||||
|
$definition = $this->getDefinition($id);
|
||||||
|
if ($definition != null) {
|
||||||
|
|
||||||
|
// Core Modules couldn´t disabled
|
||||||
|
if (!$definition['isCoreModule']) {
|
||||||
|
|
||||||
|
if (isset($definition['userModules']) && is_array($definition['userModules'])) {
|
||||||
|
$modulesToDisable = array_keys($definition['userModules']);
|
||||||
|
foreach (User::model()->findAll() as $user) {
|
||||||
|
foreach ($modulesToDisable as $userModuleId) {
|
||||||
|
if ($user->isModuleEnabled($userModuleId))
|
||||||
|
$user->uninstallModule($userModuleId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($definition['spaceModules']) && is_array($definition['spaceModules'])) {
|
||||||
|
$modulesToDisable = array_keys($definition['spaceModules']);
|
||||||
|
foreach (Space::model()->findAll() as $space) {
|
||||||
|
foreach ($modulesToDisable as $spaceModuleId) {
|
||||||
|
if ($space->isModuleEnabled($spaceModuleId))
|
||||||
|
$space->uninstallModule($spaceModuleId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get Enabled Module Record
|
||||||
|
$moduleEnabled = ModuleEnabled::model()->findByPk($id);
|
||||||
|
if ($moduleEnabled != null)
|
||||||
|
$moduleEnabled->delete();
|
||||||
|
|
||||||
|
// Fire Event Disabled Event
|
||||||
|
if ($this->hasEventHandler('onDisable'))
|
||||||
|
$this->onDisable(new CEvent($this, $id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$this->flushCache();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This event is raised after disabling a module
|
||||||
|
*
|
||||||
|
* @param CEvent $event the event parameter
|
||||||
|
* @see disable
|
||||||
|
*/
|
||||||
|
public function onDisable($event) {
|
||||||
|
$this->raiseEvent('onDisable', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This event is raised after enabling a module
|
||||||
|
*
|
||||||
|
* @param CEvent $event the event parameter
|
||||||
|
* @see enable
|
||||||
|
*/
|
||||||
|
public function onEnable($event) {
|
||||||
|
$this->raiseEvent('onEnable', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the definition array of a registered module
|
||||||
|
*
|
||||||
|
* @param type $id
|
||||||
|
* @return null
|
||||||
|
*/
|
||||||
|
public function getDefinition($id) {
|
||||||
|
if (isset($this->registeredModules[$id]))
|
||||||
|
return $this->registeredModules[$id];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a new Content Model
|
||||||
|
*
|
||||||
|
* @param String $className
|
||||||
|
*/
|
||||||
|
public function registerContentModel($className) {
|
||||||
|
$this->registeredContentModels[] = $className;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a list of all registered content models
|
||||||
|
*
|
||||||
|
* @return Array
|
||||||
|
*/
|
||||||
|
public function getContentModels() {
|
||||||
|
return $this->registeredContentModels;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
77
protected/components/WebApplication.php
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* WebApplication is the base WebApplication class
|
||||||
|
*
|
||||||
|
* @author Lucas Bartholemy <lucas@bartholemy.com>
|
||||||
|
* @package humhub.components
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class WebApplication extends CWebApplication {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inits the webapplication
|
||||||
|
*
|
||||||
|
* Also start interceptor to add event support
|
||||||
|
*/
|
||||||
|
protected function init() {
|
||||||
|
|
||||||
|
parent::init();
|
||||||
|
|
||||||
|
$this->interceptor->start();
|
||||||
|
$this->interceptor->intercept($this);
|
||||||
|
|
||||||
|
$this->moduleManager->start();
|
||||||
|
|
||||||
|
if ($this->hasEventHandler('onInit'))
|
||||||
|
$this->onInit(new CEvent($this));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This event is raised after the init is executed.
|
||||||
|
* @param CEvent $event the event parameter
|
||||||
|
*/
|
||||||
|
public function onInit($event) {
|
||||||
|
$this->raiseEvent('onInit', $event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an array of available language codes
|
||||||
|
*
|
||||||
|
* @return Array
|
||||||
|
*/
|
||||||
|
public static function getLanguages() {
|
||||||
|
|
||||||
|
$languages = array();
|
||||||
|
$languages['en'] = 'en';
|
||||||
|
|
||||||
|
$files = scandir(Yii::app()->basePath . '/messages');
|
||||||
|
foreach ($files as $file) {
|
||||||
|
if ($file == '.' || $file == '..')
|
||||||
|
continue;
|
||||||
|
$languages[$file] = $file;
|
||||||
|
}
|
||||||
|
return $languages;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
81
protected/config/_defaults.php
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'basePath' => dirname(__FILE__) . DIRECTORY_SEPARATOR . '..',
|
||||||
|
// Default Application Name
|
||||||
|
'name' => 'HumHub',
|
||||||
|
'preload' => array('log', 'input'),
|
||||||
|
'components' => array(
|
||||||
|
// Database
|
||||||
|
'db' => array(
|
||||||
|
'connectionString' => 'mysql:host=localhost;dbname=test',
|
||||||
|
'emulatePrepare' => true,
|
||||||
|
'charset' => 'utf8',
|
||||||
|
'enableProfiling' => true,
|
||||||
|
'enableParamLogging' => true,
|
||||||
|
'schemaCachingDuration' => 3600,
|
||||||
|
),
|
||||||
|
'moduleManager' => array(
|
||||||
|
'class' => 'application.components.ModuleManager',
|
||||||
|
),
|
||||||
|
'messages' => array(
|
||||||
|
'class' => 'application.components.HPhpMessageSource',
|
||||||
|
),
|
||||||
|
'input' => array(
|
||||||
|
'class' => 'application.extensions.CmsInput',
|
||||||
|
'cleanPost' => false,
|
||||||
|
'cleanGet' => false,
|
||||||
|
),
|
||||||
|
'interceptor' => array(
|
||||||
|
'class' => 'HInterceptor',
|
||||||
|
),
|
||||||
|
'session' => array(
|
||||||
|
'class' => 'application.modules_core.user.components.SIHttpSession',
|
||||||
|
'connectionID' => 'db',
|
||||||
|
'sessionName' => 'sin',
|
||||||
|
),
|
||||||
|
'request' => array(
|
||||||
|
//'enableCsrfValidation' => true,
|
||||||
|
),
|
||||||
|
// Caching (Will replaced at runtime)
|
||||||
|
'cache' => array(
|
||||||
|
'class' => 'CDummyCache'
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// Modules
|
||||||
|
'modules' => array(
|
||||||
|
// All HumHub Modules will automatically loaded via
|
||||||
|
// /modules/*/autostart.php
|
||||||
|
// or
|
||||||
|
// /modules_core/*/autostart.php
|
||||||
|
),
|
||||||
|
// autoloading model and component classes
|
||||||
|
'import' => array(
|
||||||
|
'application.models.*',
|
||||||
|
'application.forms.*',
|
||||||
|
'application.components.*',
|
||||||
|
'application.behaviors.*',
|
||||||
|
'application.interfaces.*',
|
||||||
|
'application.libs.*',
|
||||||
|
'application.widgets.*',
|
||||||
|
// 3rd Party Extensions
|
||||||
|
'ext.yii-mail.YiiMailMessage',
|
||||||
|
'ext.EZendAutoloader.EZendAutoloader',
|
||||||
|
),
|
||||||
|
// application-level parameters that can be accessed
|
||||||
|
// using Yii::app()->params['paramName']
|
||||||
|
'params' => array(
|
||||||
|
// Installed Flag
|
||||||
|
'installed' => false,
|
||||||
|
// Authentication Options
|
||||||
|
'auth' => array(
|
||||||
|
// Available Registration Types
|
||||||
|
'modes' => array('local'),
|
||||||
|
// Local Authentication Options
|
||||||
|
'local' => array(
|
||||||
|
'freeRegistration' => true,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
?>
|
3
protected/config/_settings.php
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<?php return array (
|
||||||
|
|
||||||
|
); ?>
|
66
protected/config/console.php
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Console Default Configuration
|
||||||
|
*
|
||||||
|
* This configuration file only affects only the console application.
|
||||||
|
*/
|
||||||
|
$pre_config = CMap::mergeArray(require (dirname(__FILE__) . '/_defaults.php'), require (dirname(__FILE__) . '/_settings.php'));
|
||||||
|
return CMap::mergeArray($pre_config, array(
|
||||||
|
'preload' => array('log'),
|
||||||
|
'behaviors' => array(
|
||||||
|
'viewRenderer' => 'HConsoleApplicationBehavior',
|
||||||
|
),
|
||||||
|
'components' => array(
|
||||||
|
'urlManager' => array(
|
||||||
|
'urlFormat' => 'path',
|
||||||
|
'showScriptName' => false,
|
||||||
|
'rules' => array(
|
||||||
|
'<controller:\w+>/<id:\d+>' => '<controller>/view',
|
||||||
|
'<controller:\w+>/<action:\w+>/<id:\d+>' => '<controller>/<action>',
|
||||||
|
'<controller:\w+>/<action:\w+>' => '<controller>/<action>',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'request' => array(
|
||||||
|
'class' => 'HHttpRequest',
|
||||||
|
),
|
||||||
|
'user' => array(
|
||||||
|
'class' => 'ConsoleUser',
|
||||||
|
),
|
||||||
|
'log' => array(
|
||||||
|
'class' => 'CLogRouter',
|
||||||
|
'routes' => array(
|
||||||
|
array(
|
||||||
|
'class' => 'CFileLogRoute',
|
||||||
|
'levels' => 'error, warning',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// autoloading model and component classes
|
||||||
|
'import' => array(
|
||||||
|
'application.models.*',
|
||||||
|
'application.forms.*',
|
||||||
|
'application.components.*',
|
||||||
|
'application.vendors.yii.cli.commands.*',
|
||||||
|
'application.libs.*',
|
||||||
|
'application.modules_core.user.components.*',
|
||||||
|
'ext.EZendAutoloader.EZendAutoloader',
|
||||||
|
'ext.migrate-command.*',
|
||||||
|
),
|
||||||
|
'commandMap' => array(
|
||||||
|
'message' => 'application.commands.shell.ZMessageCommand',
|
||||||
|
'search_rebuild' => 'application.commands.shell.SearchIndexer.Rebuilder',
|
||||||
|
'search_optimize' => 'application.commands.shell.SearchIndexer.Optimize',
|
||||||
|
'integritychecker' => 'application.commands.shell.Maintain.IntegrityChecker',
|
||||||
|
'search' => 'application.modules.zstunden.commands.Search',
|
||||||
|
'emailing' => 'application.commands.shell.EMailing.EMailing',
|
||||||
|
'emailing_test' => 'application.commands.shell.EMailing.TestMail',
|
||||||
|
'cron' => 'application.commands.shell.ZCron.ZCronRunner',
|
||||||
|
'migrate' => array(
|
||||||
|
'class' => 'application.commands.shell.ZMigrateCommand',
|
||||||
|
'migrationPath' => 'application.migrations',
|
||||||
|
'migrationTable' => 'migration',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
));
|
46
protected/config/htmlpurifier.php
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
<?php if ( ! defined('YII_PATH')) exit('No direct script access allowed');
|
||||||
|
|
||||||
|
return array(
|
||||||
|
|
||||||
|
/*'Attr.AllowedRel' => array('noindex','nofollow'),
|
||||||
|
'Attr.DefaultImageAlt' => NULL,
|
||||||
|
'Core.ColorKeywords' => array(
|
||||||
|
'maroon' => '#800000',
|
||||||
|
'red' => '#FF0000',
|
||||||
|
'orange' => '#FFA500',
|
||||||
|
'yellow' => '#FFFF00',
|
||||||
|
'olive' => '#808000',
|
||||||
|
'purple' => '#800080',
|
||||||
|
'fuchsia' => '#FF00FF',
|
||||||
|
'white' => '#FFFFFF',
|
||||||
|
'lime' => '#00FF00',
|
||||||
|
'green' => '#008000',
|
||||||
|
'navy' => '#000080',
|
||||||
|
'blue' => '#0000FF',
|
||||||
|
'aqua' => '#00FFFF',
|
||||||
|
'teal' => '#008080',
|
||||||
|
'black' => '#000000',
|
||||||
|
'silver' => '#C0C0C0',
|
||||||
|
'gray' => '#808080',
|
||||||
|
),
|
||||||
|
'Core.Encoding' => Yii::app()->charset,
|
||||||
|
'Core.EscapeInvalidTags' => FALSE,
|
||||||
|
'HTML.AllowedElements' => array(
|
||||||
|
'a','b','em','small','strong','del','q','img','span','ul','ol','li','h1','h2','h3','h4','h5','h6'
|
||||||
|
),
|
||||||
|
'HTML.AllowedAttributes' => array(
|
||||||
|
'href','rel','target','src', 'style',
|
||||||
|
),
|
||||||
|
*/
|
||||||
|
'HTML.Doctype' => 'XHTML 1.0 Transitional',
|
||||||
|
'URI.AllowedSchemes' => array(
|
||||||
|
'http' => true,
|
||||||
|
'https' => true,
|
||||||
|
'mailto' => true,
|
||||||
|
'ftp' => true,
|
||||||
|
'nntp' => true,
|
||||||
|
'news' => true,
|
||||||
|
),
|
||||||
|
'URI.Base'=>NULL,
|
||||||
|
|
||||||
|
);
|
27
protected/config/i18n.php
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* This is the configuration for generating message translations
|
||||||
|
* for the Yii framework. It is used by the 'yiic message' command.
|
||||||
|
*/
|
||||||
|
return array(
|
||||||
|
//'sourcePath'=>dirname(__FILE__).DIRECTORY_SEPARATOR.'../..',
|
||||||
|
'sourcePath'=>dirname(__FILE__).DIRECTORY_SEPARATOR.'..',
|
||||||
|
|
||||||
|
'messagePath'=>dirname(__FILE__).DIRECTORY_SEPARATOR.'..'.DIRECTORY_SEPARATOR.'messages',
|
||||||
|
//'languages'=>array('fi','zh_cn','zh_tw','de','de_du','1337', 'el','es','sv','he','nl','pt','pt_br','ru','it','fr','ja','pl','hu','ro','id','vi','bg','lv','sk','uk','ko_kr','kk','cs'),
|
||||||
|
'languages'=>array('de','fr'),
|
||||||
|
|
||||||
|
'fileTypes'=>array('php'),
|
||||||
|
'overwrite'=>true,
|
||||||
|
'exclude'=>array(
|
||||||
|
'.svn',
|
||||||
|
'.gitignore',
|
||||||
|
'yiilite.php',
|
||||||
|
'yiit.php',
|
||||||
|
'/i18n/data',
|
||||||
|
'/messages',
|
||||||
|
'/vendors',
|
||||||
|
'/web/js',
|
||||||
|
'yii',
|
||||||
|
),
|
||||||
|
);
|
61
protected/config/main.php
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Web Application Default Configuration
|
||||||
|
*
|
||||||
|
* This file not affects console applications!
|
||||||
|
*/
|
||||||
|
Yii::setPathOfAlias('modules_core', dirname(__FILE__) . '/../modules_core');
|
||||||
|
|
||||||
|
|
||||||
|
$pre_config = CMap::mergeArray(require (dirname(__FILE__) . '/_defaults.php'), require (dirname(__FILE__) . '/_settings.php'));
|
||||||
|
|
||||||
|
return CMap::mergeArray($pre_config, array(
|
||||||
|
// preloading 'log' component
|
||||||
|
'preload' => array('log'),
|
||||||
|
// application components
|
||||||
|
'components' => array(
|
||||||
|
// Session specific settings
|
||||||
|
'session' => array(
|
||||||
|
),
|
||||||
|
'user' => array(
|
||||||
|
// enable cookie-based authentication
|
||||||
|
'allowAutoLogin' => true,
|
||||||
|
'class' => 'application.modules_core.user.components.WebUser',
|
||||||
|
'loginUrl' => array('//user/auth/login'),
|
||||||
|
),
|
||||||
|
'request' => array(
|
||||||
|
'class' => 'HHttpRequest',
|
||||||
|
'enableCsrfValidation' => true,
|
||||||
|
),
|
||||||
|
'clientScript' => array(
|
||||||
|
'class' => 'HClientScript',
|
||||||
|
),
|
||||||
|
// uncomment the following to enable URLs in path-format
|
||||||
|
'urlManager' => array(
|
||||||
|
'urlFormat' => 'path',
|
||||||
|
'showScriptName' => false,
|
||||||
|
'rules' => array(
|
||||||
|
'<controller:\w+>/<id:\d+>' => '<controller>/view',
|
||||||
|
'<controller:\w+>/<action:\w+>/<id:\d+>' => '<controller>/<action>',
|
||||||
|
'<controller:\w+>/<action:\w+>' => '<controller>/<action>',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'themeManager' => array(
|
||||||
|
'themeClass' => 'HTheme',
|
||||||
|
),
|
||||||
|
'errorHandler' => array(
|
||||||
|
// use 'site/error' action to display errors
|
||||||
|
'errorAction' => '//site/error',
|
||||||
|
),
|
||||||
|
'log' => array(
|
||||||
|
'class' => 'CLogRouter',
|
||||||
|
'routes' => array(
|
||||||
|
array(
|
||||||
|
'class' => 'CFileLogRoute',
|
||||||
|
'levels' => 'error, warning',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
));
|
182
protected/controllers/DashboardController.php
Normal file
@ -0,0 +1,182 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard Controller
|
||||||
|
*
|
||||||
|
* @package humhub.controllers
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class DashboardController extends Controller {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array action filters
|
||||||
|
*/
|
||||||
|
public function filters() {
|
||||||
|
return array(
|
||||||
|
'accessControl', // perform access control for CRUD operations
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies the access control rules.
|
||||||
|
* This method is used by the 'accessControl' filter.
|
||||||
|
* @return array access control rules
|
||||||
|
*/
|
||||||
|
public function accessRules() {
|
||||||
|
return array(
|
||||||
|
array('allow', // allow authenticated user to perform 'create' and 'update' actions
|
||||||
|
'users' => array('@'),
|
||||||
|
),
|
||||||
|
array('deny', // deny all users
|
||||||
|
'users' => array('*'),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dashboard Index
|
||||||
|
*
|
||||||
|
* Show recent wall entries for this user
|
||||||
|
*/
|
||||||
|
public function actionIndex() {
|
||||||
|
|
||||||
|
// contains the current version to show the welcome modal
|
||||||
|
$version = 1;
|
||||||
|
|
||||||
|
Yii::app()->user->model->checkWall();
|
||||||
|
|
||||||
|
//$mySpaces = Yii::app()->user->getModel()->getSpaces();
|
||||||
|
//$this->render('index', array('mySpaces' => $mySpaces));
|
||||||
|
$this->render('index', array('version' => $version));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Page which shows all current workspaces
|
||||||
|
*/
|
||||||
|
public function actionSpaces() {
|
||||||
|
|
||||||
|
$pageSize = 5;
|
||||||
|
|
||||||
|
$criteria = new CDbCriteria();
|
||||||
|
$criteria->condition = 'visibility != ' . Space::VISIBILITY_NONE;
|
||||||
|
$criteria->order = 'id DESC';
|
||||||
|
//$criteria->params = array (':id'=>$id);
|
||||||
|
|
||||||
|
$workspaceCount = Space::model()->count($criteria);
|
||||||
|
|
||||||
|
$pages = new CPagination($workspaceCount);
|
||||||
|
$pages->setPageSize($pageSize);
|
||||||
|
$pages->applyLimit($criteria); // the trick is here!
|
||||||
|
|
||||||
|
$workspaces = Space::model()->findAll($criteria);
|
||||||
|
|
||||||
|
$this->render('workspaces', array(
|
||||||
|
'workspaces' => $workspaces,
|
||||||
|
'pages' => $pages,
|
||||||
|
'workspaceCount' => $workspaceCount,
|
||||||
|
'pageSize' => $pageSize,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Page which shows all people
|
||||||
|
*/
|
||||||
|
public function actionPeople() {
|
||||||
|
|
||||||
|
$criteria = new CDbCriteria();
|
||||||
|
//$criteria->condition = 'visibility != '.Space::VISIBILITY_NONE;
|
||||||
|
$criteria->order = 'firstname ASC';
|
||||||
|
|
||||||
|
$userCount = User::model()->count($criteria);
|
||||||
|
|
||||||
|
$pages = new CPagination($userCount);
|
||||||
|
$pages->setPageSize(HSetting::Get('paginationSize'));
|
||||||
|
$pages->applyLimit($criteria); // the trick is here!
|
||||||
|
|
||||||
|
$users = User::model()->findAll($criteria);
|
||||||
|
|
||||||
|
$this->render('people', array(
|
||||||
|
'users' => $users,
|
||||||
|
'pages' => $pages,
|
||||||
|
'userCount' => $userCount,
|
||||||
|
'pageSize' => HSetting::Get('paginationSize'),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a JSON Object which contains a lot of informations about
|
||||||
|
* current states like new posts on workspaces
|
||||||
|
*/
|
||||||
|
public function actionGetFrontEndInfo() {
|
||||||
|
|
||||||
|
$json = array();
|
||||||
|
$json['workspaces'] = array();
|
||||||
|
|
||||||
|
$criteria = new CDbCriteria();
|
||||||
|
$criteria->order = 'last_visit DESC';
|
||||||
|
|
||||||
|
$memberships = UserSpaceMembership::model()->with('workspace')->findAllByAttributes(array(
|
||||||
|
'user_id' => Yii::app()->user->id,
|
||||||
|
'status' => UserSpaceMembership::STATUS_MEMBER
|
||||||
|
), $criteria);
|
||||||
|
|
||||||
|
foreach ($memberships as $membership) {
|
||||||
|
$workspace = $membership->workspace;
|
||||||
|
|
||||||
|
$info = array();
|
||||||
|
$info['name'] = $workspace->name;
|
||||||
|
#$info['id'] = $workspace->id; # should be hidden at frontend
|
||||||
|
$info['guid'] = $workspace->guid;
|
||||||
|
$info['totalItems'] = $workspace->countItems();
|
||||||
|
$info['newItems'] = $membership->countNewItems();
|
||||||
|
|
||||||
|
$json['workspaces'][] = $info;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Yii::app()->moduleManager->isEnabled('mail')) {
|
||||||
|
// New message count
|
||||||
|
$sql = "SELECT count(message_id)
|
||||||
|
FROM user_message
|
||||||
|
LEFT JOIN message on message.id = user_message.message_id
|
||||||
|
WHERE user_message.user_id = :user_id AND (message.updated_at > user_message.last_viewed OR user_message.last_viewed IS NULL)";
|
||||||
|
$connection = Yii::app()->db;
|
||||||
|
$command = $connection->createCommand($sql);
|
||||||
|
$userId = Yii::app()->user->id;
|
||||||
|
$command->bindParam(":user_id", $userId);
|
||||||
|
$json['newMessages'] = $command->queryScalar();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// New notification count
|
||||||
|
$sql = "SELECT count(id)
|
||||||
|
FROM notification
|
||||||
|
WHERE user_id = :user_id AND seen != 1";
|
||||||
|
$connection = Yii::app()->db;
|
||||||
|
$command = $connection->createCommand($sql);
|
||||||
|
$userId = Yii::app()->user->id;
|
||||||
|
$command->bindParam(":user_id", $userId);
|
||||||
|
$json['newNotifications'] = $command->queryScalar();
|
||||||
|
|
||||||
|
print CJSON::encode($json);
|
||||||
|
Yii::app()->end();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
156
protected/controllers/SearchController.php
Normal file
@ -0,0 +1,156 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search Controller provides search functions inside the application.
|
||||||
|
*
|
||||||
|
* @author Luke
|
||||||
|
* @package humhub.controllers
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class SearchController extends Controller {
|
||||||
|
|
||||||
|
public $subLayout = "_layout";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array action filters
|
||||||
|
*/
|
||||||
|
public function filters() {
|
||||||
|
return array(
|
||||||
|
'accessControl', // perform access control for CRUD operations
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies the access control rules.
|
||||||
|
* This method is used by the 'accessControl' filter.
|
||||||
|
* @return array access control rules
|
||||||
|
*/
|
||||||
|
public function accessRules() {
|
||||||
|
return array(
|
||||||
|
array('allow', // allow authenticated user to perform 'create' and 'update' actions
|
||||||
|
'users' => array('@'),
|
||||||
|
),
|
||||||
|
array('deny', // deny all users
|
||||||
|
'users' => array('*'),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SearchAction
|
||||||
|
*
|
||||||
|
* Modes: normal for full page, quick as partial for lightbox
|
||||||
|
*/
|
||||||
|
public function actionIndex() {
|
||||||
|
|
||||||
|
// Get Parameters
|
||||||
|
$keyword = Yii::app()->request->getParam('keyword', "");
|
||||||
|
$spaceGuid = Yii::app()->request->getParam('sguid', "");
|
||||||
|
$mode = Yii::app()->request->getParam('mode', "normal");
|
||||||
|
$page = (int) Yii::app()->request->getParam('page', 1); // current page (pagination)
|
||||||
|
// Cleanup
|
||||||
|
$keyword = Yii::app()->input->stripClean($keyword);
|
||||||
|
$spaceGuid = Yii::app()->input->stripClean($spaceGuid);
|
||||||
|
|
||||||
|
if ($mode != 'quick') {
|
||||||
|
$mode = "normal";
|
||||||
|
}
|
||||||
|
|
||||||
|
$limit = HSetting::Get('paginationSize'); // Show Hits
|
||||||
|
$hitCount = 0; // Total Hit Count
|
||||||
|
$query = ""; // Lucene Query
|
||||||
|
$append = " AND (model:User OR model:Space)"; // Appends for Lucene Query
|
||||||
|
$moreResults = false; // Indicates if there are more hits
|
||||||
|
$results = array();
|
||||||
|
|
||||||
|
// Quick Search shows always 1
|
||||||
|
if ($mode == 'quick')
|
||||||
|
$limit = 5;
|
||||||
|
|
||||||
|
// Load also Space if requested
|
||||||
|
$currentSpace = null;
|
||||||
|
if ($spaceGuid) {
|
||||||
|
$currentSpace = Space::model()->findByAttributes(array('guid' => $spaceGuid));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* $index = new Zend_Search_Lucene_Interface_MultiSearcher();
|
||||||
|
* $index->addIndex(Zend_Search_Lucene::open('search/index1'));
|
||||||
|
* $index->addIndex(Zend_Search_Lucene::open('search/index2'));
|
||||||
|
* $index->find('someSearchQuery');
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Do Search
|
||||||
|
if ($keyword != "") {
|
||||||
|
|
||||||
|
if ($currentSpace != null) {
|
||||||
|
$append = " AND (model:User OR model:Space OR (belongsToType:Space AND belongsToId:" . $currentSpace->id . "))";
|
||||||
|
}
|
||||||
|
|
||||||
|
$hits = new ArrayObject(HSearch::getInstance()->Find($keyword . $append));
|
||||||
|
$hitCount = count($hits);
|
||||||
|
|
||||||
|
// Limit Hits
|
||||||
|
$hits = new LimitIterator($hits->getIterator(), ($page - 1) * $limit, $limit);
|
||||||
|
|
||||||
|
if ($hitCount > $limit)
|
||||||
|
$moreResults = true;
|
||||||
|
|
||||||
|
// Build Results Array
|
||||||
|
|
||||||
|
foreach ($hits as $hit) {
|
||||||
|
|
||||||
|
$doc = $hit->getDocument();
|
||||||
|
$model = $doc->getField('model')->value;
|
||||||
|
$pk = $doc->getField('pk')->value;
|
||||||
|
|
||||||
|
$object = $model::model()->findByPk($pk);
|
||||||
|
$results[] = $object->getSearchResult();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create Pagination Class
|
||||||
|
$pages = new CPagination($hitCount);
|
||||||
|
$pages->setPageSize($limit);
|
||||||
|
$_GET['keyword'] = $keyword; // Fix for post var
|
||||||
|
|
||||||
|
if ($mode == 'quick') {
|
||||||
|
$this->renderPartial('quick', array(
|
||||||
|
'keyword' => $keyword,
|
||||||
|
'results' => $results,
|
||||||
|
'spaceGuid' => $spaceGuid,
|
||||||
|
'moreResults' => $moreResults,
|
||||||
|
'hitCount' => $hitCount,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
$this->render('index', array(
|
||||||
|
'keyword' => $keyword,
|
||||||
|
'results' => $results,
|
||||||
|
'spaceGuid' => $spaceGuid,
|
||||||
|
'moreResults' => $moreResults,
|
||||||
|
'pages' => $pages, // CPagination,
|
||||||
|
'pageSize' => $limit,
|
||||||
|
'hitCount' => $hitCount,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
50
protected/controllers/SiteController.php
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HumHub
|
||||||
|
* Copyright © 2014 The HumHub Project
|
||||||
|
*
|
||||||
|
* The texts of the GNU Affero General Public License with an additional
|
||||||
|
* permission and of our proprietary license can be found at and
|
||||||
|
* in the LICENSE file you have received along with this program.
|
||||||
|
*
|
||||||
|
* According to our dual licensing model, this program can be used either
|
||||||
|
* under the terms of the GNU Affero General Public License, version 3,
|
||||||
|
* or under a proprietary license.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Site Controller
|
||||||
|
*
|
||||||
|
* @package humhub.controllers
|
||||||
|
* @since 0.5
|
||||||
|
*/
|
||||||
|
class SiteController extends Controller {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is the default 'index' action that is invoked
|
||||||
|
* when an action is not explicitly requested by users.
|
||||||
|
*/
|
||||||
|
public function actionIndex() {
|
||||||
|
|
||||||
|
$this->redirect(array('dashboard/index'));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is the action to handle external exceptions.
|
||||||
|
*/
|
||||||
|
public function actionError() {
|
||||||
|
if ($error = Yii::app()->errorHandler->error) {
|
||||||
|
if (Yii::app()->request->isAjaxRequest)
|
||||||
|
echo $error['message'];
|
||||||
|
else
|
||||||
|
$this->render('error', $error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
BIN
protected/docs/database/database_model_old.mwb
Normal file
39
protected/docs/database/inital_data.sql
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
-- phpMyAdmin SQL Dump
|
||||||
|
-- version 4.0.0-rc2
|
||||||
|
-- http://www.phpmyadmin.net
|
||||||
|
--
|
||||||
|
-- Host: localhost
|
||||||
|
-- Erstellungszeit: 21. Sep 2013 um 06:31
|
||||||
|
-- Server Version: 5.1.66-0+squeeze1
|
||||||
|
-- PHP-Version: 5.3.3-7+squeeze15
|
||||||
|
|
||||||
|
SET SQL_MODE = "NO_AUTO_VALUE_ON_ZERO";
|
||||||
|
SET time_zone = "+00:00";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Datenbank: `zamm_clean`
|
||||||
|
--
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Daten für Tabelle `group`
|
||||||
|
--
|
||||||
|
|
||||||
|
INSERT INTO `group` (`id`, `space_id`, `name`, `description`, `created_at`, `created_by`, `updated_at`, `updated_by`) VALUES
|
||||||
|
(1, NULL, 'Example Group', '', '2012-10-16 01:06:19', 1, '2012-10-16 01:06:19', 1);
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Daten für Tabelle `registry`
|
||||||
|
--
|
||||||
|
|
||||||
|
INSERT INTO `registry` (`id`, `name`, `value`, `value_text`, `module_id`, `created_at`, `created_by`, `updated_at`, `updated_by`) VALUES
|
||||||
|
(1, 'dbVersion', '20', NULL, NULL, '0000-00-00 00:00:00', 0, '2013-06-08 15:15:45', 0);
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Daten für Tabelle `user`
|
||||||
|
--
|
||||||
|
|
||||||
|
INSERT INTO `user` (`id`, `guid`, `user_invite_id`, `wall_id`, `group_id`, `status`, `super_admin`, `username`, `email`, `password`, `auth_mode`, `firstname`, `lastname`, `title`, `street`, `zip`, `city`, `country`, `state`, `about`, `tags`, `phone_private`, `phone_work`, `mobile`, `fax`, `language`, `im_skype`, `im_msn`, `im_icq`, `im_xmpp`, `url`, `url_facebook`, `url_linkedin`, `url_xing`, `url_youtube`, `url_vimeo`, `url_flickr`, `url_myspace`, `url_googleplus`, `url_twitter`, `receive_email_notifications`, `receive_email_messaging`, `receive_email_activities`, `last_activity_email`, `created_at`, `created_by`, `updated_at`, `updated_by`) VALUES
|
||||||
|
(1, 'b2fc71a5-abe6-adad-805f-588de32036f4', NULL, 2, 1, 1, 1, 'admin', 'l.bartholemy@zeros.ones.de', '___enc___08a593a5bec48fa3eb0c1a697968638527e83f81', 'local', 'System', 'Admin', '', '', '', 'Server', 'Germany', '', NULL, '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', 3, 3, 3, '2013-06-17 07:05:01', '2012-10-16 01:05:08', NULL, '2013-06-17 05:22:54', 1);
|
||||||
|
|
625
protected/docs/database/latest_database_structure.sql
Normal file
@ -0,0 +1,625 @@
|
|||||||
|
SET SQL_MODE = "NO_AUTO_VALUE_ON_ZERO";
|
||||||
|
SET time_zone = "+00:00";
|
||||||
|
|
||||||
|
/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */;
|
||||||
|
/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */;
|
||||||
|
/*!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */;
|
||||||
|
/*!40101 SET NAMES utf8 */;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `activity` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`type` varchar(45) DEFAULT NULL COMMENT 'CLASS_TYPE of Activity Object',
|
||||||
|
`module` varchar(100) NOT NULL,
|
||||||
|
`object_model` varchar(100) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `comment` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`message` text,
|
||||||
|
`object_model` varchar(100) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`space_id` int(11) DEFAULT NULL COMMENT 'If inside a workspace (for Counting)',
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `object_id` (`object_id`,`object_model`),
|
||||||
|
KEY `workspace_id` (`space_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `content` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`guid` varchar(45) NOT NULL,
|
||||||
|
`object_model` varchar(100) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`visibility` tinyint(4) DEFAULT NULL,
|
||||||
|
`sticked` tinyint(4) DEFAULT NULL,
|
||||||
|
`archived` tinytext,
|
||||||
|
`space_id` int(11) DEFAULT NULL,
|
||||||
|
`user_id` int(11) DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `object_model` (`object_model`,`object_id`),
|
||||||
|
UNIQUE KEY `guid` (`guid`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=latin1;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `file` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`guid` varchar(45) DEFAULT NULL,
|
||||||
|
`object_model` varchar(100) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`file_name` varchar(255) DEFAULT NULL,
|
||||||
|
`title` varchar(255) DEFAULT NULL,
|
||||||
|
`mime_type` varchar(150) DEFAULT NULL,
|
||||||
|
`size` varchar(45) DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `guid_UNIQUE` (`guid`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `group` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`space_id` int(10) DEFAULT NULL,
|
||||||
|
`name` varchar(45) DEFAULT NULL,
|
||||||
|
`description` text,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `group_admin` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`group_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=latin1;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `like` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`target_user_id` int(11) NOT NULL,
|
||||||
|
`object_model` varchar(100) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `object_id_2` (`object_id`,`object_model`,`created_by`),
|
||||||
|
KEY `object_id` (`object_id`,`object_model`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `message` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`title` varchar(255) DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `message_entry` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`message_id` int(11) NOT NULL,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`file_id` int(11) DEFAULT NULL,
|
||||||
|
`content` text NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `fk_message_content_message1` (`message_id`),
|
||||||
|
KEY `fk_message_content_user1` (`user_id`),
|
||||||
|
KEY `fk_message_entry_file1` (`file_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `migration` (
|
||||||
|
`version` varchar(255) NOT NULL,
|
||||||
|
`apply_time` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`version`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `module_enabled` (
|
||||||
|
`module_id` varchar(100) NOT NULL,
|
||||||
|
PRIMARY KEY (`module_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `note` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`title` varchar(255) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `notification` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`class` varchar(100) NOT NULL,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`seen` tinyint(4) DEFAULT NULL,
|
||||||
|
`source_object_model` varchar(100) DEFAULT NULL,
|
||||||
|
`source_object_id` int(11) DEFAULT NULL,
|
||||||
|
`target_object_model` varchar(100) DEFAULT NULL,
|
||||||
|
`target_object_id` int(11) DEFAULT NULL,
|
||||||
|
`space_id` int(11) DEFAULT NULL,
|
||||||
|
`emailed` tinyint(4) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `poll` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`question` varchar(255) NOT NULL,
|
||||||
|
`allow_multiple` tinyint(4) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `poll_answer` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`poll_id` int(11) NOT NULL,
|
||||||
|
`answer` varchar(255) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `doodle_id` (`poll_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `poll_answer_user` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`poll_id` int(11) NOT NULL,
|
||||||
|
`poll_answer_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `doodle_answer_id_2` (`poll_answer_id`,`created_by`),
|
||||||
|
KEY `doodle_id` (`poll_id`),
|
||||||
|
KEY `doodle_answer_id` (`poll_answer_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `post` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`message` text,
|
||||||
|
`original_message` text,
|
||||||
|
`url` varchar(255) DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `profile` (
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`firstname` varchar(255) DEFAULT NULL,
|
||||||
|
`lastname` varchar(255) DEFAULT NULL,
|
||||||
|
`title` varchar(255) DEFAULT NULL,
|
||||||
|
`street` varchar(255) DEFAULT NULL,
|
||||||
|
`zip` int(11) DEFAULT NULL,
|
||||||
|
`city` varchar(255) DEFAULT NULL,
|
||||||
|
`country` varchar(255) DEFAULT NULL,
|
||||||
|
`state` varchar(255) DEFAULT NULL,
|
||||||
|
`about` text,
|
||||||
|
`phone_private` varchar(255) DEFAULT NULL,
|
||||||
|
`phone_work` varchar(255) DEFAULT NULL,
|
||||||
|
`mobile` varchar(255) DEFAULT NULL,
|
||||||
|
`fax` varchar(255) DEFAULT NULL,
|
||||||
|
`im_skype` varchar(255) DEFAULT NULL,
|
||||||
|
`im_msn` varchar(255) DEFAULT NULL,
|
||||||
|
`im_icq` int(11) DEFAULT NULL,
|
||||||
|
`im_xmpp` varchar(255) DEFAULT NULL,
|
||||||
|
`url` varchar(255) DEFAULT NULL,
|
||||||
|
`url_facebook` varchar(255) DEFAULT NULL,
|
||||||
|
`url_linkedin` varchar(255) DEFAULT NULL,
|
||||||
|
`url_xing` varchar(255) DEFAULT NULL,
|
||||||
|
`url_youtube` varchar(255) DEFAULT NULL,
|
||||||
|
`url_vimeo` varchar(255) DEFAULT NULL,
|
||||||
|
`url_flickr` varchar(255) DEFAULT NULL,
|
||||||
|
`url_myspace` varchar(255) DEFAULT NULL,
|
||||||
|
`url_googleplus` varchar(255) DEFAULT NULL,
|
||||||
|
`url_twitter` varchar(255) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`user_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `profile_field` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`profile_field_category_id` int(11) NOT NULL,
|
||||||
|
`module_id` varchar(255) DEFAULT NULL,
|
||||||
|
`field_type_class` varchar(255) NOT NULL,
|
||||||
|
`field_type_config` text,
|
||||||
|
`internal_name` varchar(100) NOT NULL,
|
||||||
|
`title` varchar(255) NOT NULL,
|
||||||
|
`description` text,
|
||||||
|
`sort_order` int(11) NOT NULL,
|
||||||
|
`required` tinyint(4) DEFAULT NULL,
|
||||||
|
`show_at_registration` tinyint(4) DEFAULT NULL,
|
||||||
|
`editable` tinyint(4) NOT NULL DEFAULT '1',
|
||||||
|
`visible` tinyint(4) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `profile_field_category` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`title` varchar(255) NOT NULL,
|
||||||
|
`description` text NOT NULL,
|
||||||
|
`sort_order` int(11) NOT NULL,
|
||||||
|
`module_id` int(11) DEFAULT NULL,
|
||||||
|
`visibility` tinyint(4) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `setting` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`name` varchar(100) NOT NULL,
|
||||||
|
`value` varchar(255) NOT NULL,
|
||||||
|
`value_text` text,
|
||||||
|
`module_id` varchar(100) DEFAULT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `name` (`name`,`module_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `sifeedback` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`message` text NOT NULL,
|
||||||
|
`username` varchar(255) NOT NULL,
|
||||||
|
`email` varchar(255) NOT NULL,
|
||||||
|
`current_url` varchar(255) NOT NULL,
|
||||||
|
`rating` varchar(255) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `space` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`guid` varchar(45) DEFAULT NULL,
|
||||||
|
`wall_id` int(11) DEFAULT NULL,
|
||||||
|
`name` varchar(45) NOT NULL,
|
||||||
|
`description` text,
|
||||||
|
`website` varchar(45) DEFAULT NULL,
|
||||||
|
`join_policy` tinyint(4) DEFAULT NULL COMMENT '0=NO; 1=application; 2=free',
|
||||||
|
`visibility` tinyint(4) DEFAULT NULL COMMENT '0=Unvisible; 1=Registered Only; 2=Always/Everybody',
|
||||||
|
`status` tinyint(4) NOT NULL DEFAULT '1' COMMENT '0=Disabled; 1=Enabled; 2=Archived',
|
||||||
|
`tags` text,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `name` (`name`),
|
||||||
|
UNIQUE KEY `guid` (`guid`),
|
||||||
|
KEY `fk_Group_Wall1` (`wall_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `space_follow` (
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`space_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`user_id`,`space_id`),
|
||||||
|
KEY `fk_user_has_workspace_workspace1` (`space_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `space_module` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`module_id` varchar(255) NOT NULL,
|
||||||
|
`space_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `task` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`title` text NOT NULL,
|
||||||
|
`deathline` datetime DEFAULT NULL,
|
||||||
|
`max_users` int(11) NOT NULL,
|
||||||
|
`status` int(11) NOT NULL,
|
||||||
|
`percent` smallint(6) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `task_user` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`task_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `user_id` (`user_id`,`task_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`guid` varchar(45) DEFAULT NULL,
|
||||||
|
`user_invite_id` int(11) DEFAULT NULL,
|
||||||
|
`wall_id` int(11) DEFAULT NULL,
|
||||||
|
`group_id` int(11) DEFAULT NULL,
|
||||||
|
`status` tinyint(4) NOT NULL DEFAULT '2' COMMENT '0=Disabled; 1=Enabled; 2=NeedApproval; 3=Deleted',
|
||||||
|
`super_admin` tinyint(4) NOT NULL DEFAULT '0' COMMENT 'Is SuperAdmin?',
|
||||||
|
`username` varchar(25) DEFAULT NULL,
|
||||||
|
`email` varchar(100) DEFAULT NULL,
|
||||||
|
`password` varchar(200) DEFAULT NULL,
|
||||||
|
`auth_mode` varchar(10) NOT NULL,
|
||||||
|
`tags` text,
|
||||||
|
`language` varchar(5) NOT NULL,
|
||||||
|
`receive_email_notifications` tinyint(4) NOT NULL,
|
||||||
|
`receive_email_messaging` tinyint(4) NOT NULL,
|
||||||
|
`receive_email_activities` tinyint(4) NOT NULL,
|
||||||
|
`last_activity_email` datetime NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `idUser_UNIQUE` (`id`),
|
||||||
|
UNIQUE KEY `email_UNIQUE` (`email`),
|
||||||
|
UNIQUE KEY `username_UNIQUE` (`username`),
|
||||||
|
UNIQUE KEY `idWall_UNIQUE` (`wall_id`),
|
||||||
|
UNIQUE KEY `idUserInvite_UNIQUE` (`user_invite_id`),
|
||||||
|
KEY `fk_user_group1` (`group_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_content` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`object_model` varchar(100) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `user_id` (`user_id`,`object_model`,`object_id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_follow` (
|
||||||
|
`user_follower_id` int(11) NOT NULL,
|
||||||
|
`user_followed_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`user_follower_id`,`user_followed_id`),
|
||||||
|
KEY `fk_UserFollows_User2` (`user_followed_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_http_session` (
|
||||||
|
`id` char(32) NOT NULL,
|
||||||
|
`expire` int(11) DEFAULT NULL,
|
||||||
|
`user_id` int(11) DEFAULT NULL,
|
||||||
|
`data` longblob,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_invite` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`user_originator_id` int(11) DEFAULT NULL,
|
||||||
|
`space_invite_id` int(11) DEFAULT NULL,
|
||||||
|
`email` varchar(45) NOT NULL,
|
||||||
|
`source` varchar(45) DEFAULT NULL COMMENT 'Source Key where invite comes from',
|
||||||
|
`token` varchar(45) DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `email_UNIQUE` (`email`),
|
||||||
|
UNIQUE KEY `token_UNIQUE` (`token`),
|
||||||
|
KEY `fk_user_invite_Group1` (`space_invite_id`),
|
||||||
|
KEY `fk_user_invite_user1` (`user_originator_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_message` (
|
||||||
|
`message_id` int(11) NOT NULL,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`is_originator` tinyint(4) DEFAULT NULL,
|
||||||
|
`last_viewed` datetime DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`message_id`,`user_id`),
|
||||||
|
KEY `fk_message_has_user_user1` (`user_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_module` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`module_id` varchar(255) NOT NULL,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_old` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`guid` varchar(45) DEFAULT NULL,
|
||||||
|
`user_invite_id` int(11) DEFAULT NULL,
|
||||||
|
`wall_id` int(11) DEFAULT NULL,
|
||||||
|
`group_id` int(11) DEFAULT NULL,
|
||||||
|
`status` tinyint(4) NOT NULL DEFAULT '2' COMMENT '0=Disabled; 1=Enabled; 2=NeedApproval; 3=Deleted',
|
||||||
|
`super_admin` tinyint(4) NOT NULL DEFAULT '0' COMMENT 'Is SuperAdmin?',
|
||||||
|
`username` varchar(25) DEFAULT NULL,
|
||||||
|
`email` varchar(100) DEFAULT NULL,
|
||||||
|
`password` varchar(200) DEFAULT NULL,
|
||||||
|
`auth_mode` varchar(10) NOT NULL,
|
||||||
|
`firstname` varchar(45) DEFAULT NULL,
|
||||||
|
`lastname` varchar(45) DEFAULT NULL,
|
||||||
|
`title` varchar(45) DEFAULT NULL COMMENT 'z.B. Kaffeetrinker',
|
||||||
|
`street` varchar(45) DEFAULT NULL,
|
||||||
|
`zip` varchar(45) DEFAULT NULL,
|
||||||
|
`city` varchar(45) DEFAULT NULL,
|
||||||
|
`country` varchar(45) DEFAULT NULL,
|
||||||
|
`state` varchar(45) DEFAULT NULL,
|
||||||
|
`about` text,
|
||||||
|
`tags` text,
|
||||||
|
`phone_private` varchar(100) NOT NULL,
|
||||||
|
`phone_work` varchar(100) NOT NULL,
|
||||||
|
`mobile` varchar(100) NOT NULL,
|
||||||
|
`fax` varchar(100) NOT NULL,
|
||||||
|
`language` varchar(5) NOT NULL,
|
||||||
|
`im_skype` varchar(100) NOT NULL,
|
||||||
|
`im_msn` varchar(100) NOT NULL,
|
||||||
|
`im_icq` varchar(100) NOT NULL,
|
||||||
|
`im_xmpp` varchar(100) NOT NULL,
|
||||||
|
`url` varchar(255) NOT NULL,
|
||||||
|
`url_facebook` varchar(255) NOT NULL,
|
||||||
|
`url_linkedin` varchar(255) NOT NULL,
|
||||||
|
`url_xing` varchar(255) NOT NULL,
|
||||||
|
`url_youtube` varchar(255) NOT NULL,
|
||||||
|
`url_vimeo` varchar(255) NOT NULL,
|
||||||
|
`url_flickr` varchar(255) NOT NULL,
|
||||||
|
`url_myspace` varchar(255) NOT NULL,
|
||||||
|
`url_googleplus` varchar(255) NOT NULL,
|
||||||
|
`url_twitter` varchar(255) NOT NULL,
|
||||||
|
`receive_email_notifications` tinyint(4) NOT NULL,
|
||||||
|
`receive_email_messaging` tinyint(4) NOT NULL,
|
||||||
|
`receive_email_activities` tinyint(4) NOT NULL,
|
||||||
|
`last_activity_email` datetime NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
UNIQUE KEY `idUser_UNIQUE` (`id`),
|
||||||
|
UNIQUE KEY `email_UNIQUE` (`email`),
|
||||||
|
UNIQUE KEY `username_UNIQUE` (`username`),
|
||||||
|
UNIQUE KEY `idWall_UNIQUE` (`wall_id`),
|
||||||
|
UNIQUE KEY `idUserInvite_UNIQUE` (`user_invite_id`),
|
||||||
|
KEY `fk_user_group1` (`group_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `user_space_membership` (
|
||||||
|
`space_id` int(11) NOT NULL,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`originator_user_id` varchar(45) DEFAULT NULL,
|
||||||
|
`status` tinyint(4) DEFAULT NULL COMMENT 'INVITED, APPLICANT, MEMBER',
|
||||||
|
`request_message` text,
|
||||||
|
`last_visit` datetime DEFAULT NULL,
|
||||||
|
`invite_role` tinyint(4) DEFAULT NULL,
|
||||||
|
`admin_role` tinyint(4) DEFAULT NULL,
|
||||||
|
`share_role` tinyint(4) DEFAULT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`space_id`,`user_id`),
|
||||||
|
KEY `fk_Group_has_User_User1` (`user_id`),
|
||||||
|
KEY `status` (`status`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `wall` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`type` varchar(45) DEFAULT NULL COMMENT 'USER, WORKSPACE',
|
||||||
|
`object_model` varchar(50) NOT NULL,
|
||||||
|
`object_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `object_model` (`object_model`,`object_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `wall_entry` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`wall_id` int(11) NOT NULL,
|
||||||
|
`content_id` int(11) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `fk_WallEntry_Wall1` (`wall_id`)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `zstunden_projekt_space` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`space_id` int(11) NOT NULL,
|
||||||
|
`projekt_nr` varchar(100) NOT NULL,
|
||||||
|
`created_at` datetime DEFAULT NULL,
|
||||||
|
`created_by` int(11) DEFAULT NULL,
|
||||||
|
`updated_at` datetime DEFAULT NULL,
|
||||||
|
`updated_by` int(11) DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `zstunden_timer` (
|
||||||
|
`id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
|
`user_id` int(11) NOT NULL,
|
||||||
|
`projekt_nr` varchar(50) NOT NULL,
|
||||||
|
`projektname` varchar(255) NOT NULL,
|
||||||
|
`unterprojekt` varchar(255) NOT NULL,
|
||||||
|
`ansprechpartner` varchar(255) NOT NULL,
|
||||||
|
`pause_since` datetime NOT NULL,
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`created_by` int(11) NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL,
|
||||||
|
`updated_by` int(11) NOT NULL,
|
||||||
|
`pause_total` int(11) NOT NULL,
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
|
||||||
|
|
||||||
|
/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
|
||||||
|
/*!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */;
|
||||||
|
/*!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */;
|
25
protected/docs/guide/administration/bundled_software.md
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
Bundled Software / Libaries
|
||||||
|
===========================
|
||||||
|
|
||||||
|
* Yii Framework (Core Framework) http://www.yiiframework.com/
|
||||||
|
* Zend Framework (Lucene Index)
|
||||||
|
* Code Igniter (Security Functions)
|
||||||
|
* ADLdap (Active Directory / Ldap) http://adldap.sourceforge.net/
|
||||||
|
* Some Yii Extensions (See /protected/extensions/)
|
||||||
|
* PHP Markdown by by Michel Fortin (ZammDoc Module)
|
||||||
|
* Yii-Mail - http://www.yiiframework.com/extension/mail/
|
||||||
|
* SwiftMailer - http://swiftmailer.org
|
||||||
|
* jQuery - http://jquery.com
|
||||||
|
* jQuery Cookie Plugin - https://github.com/carhartl/jquery-cookie
|
||||||
|
* jQuery Knob - http://anthonyterrien.com/knob/
|
||||||
|
* Twitter Bootstrap - http://getbootstrap.com/
|
||||||
|
* Bootstrap-datepicker.js - http://www.eyecon.ro/bootstrap-datepicker
|
||||||
|
* Lightbox for Bootstrap 3 - https://github.com/ashleydw/lightbox
|
||||||
|
* Modernizr - http://modernizr.com/
|
||||||
|
* HTML5 Shiv - https://github.com/aFarkas/html5shiv
|
||||||
|
* Wysihtml5 - https://github.com/xing/wysihtml5
|
||||||
|
* Bootstrap-wysihtml5 - http://jhollingworth.github.io/bootstrap-wysihtml5/
|
||||||
|
* Autosize - http://www.jacklmoore.com/autosize
|
||||||
|
* Highlight - http://johannburkard.de/blog/programming/javascript/highlight-javascript-text-higlighting-jquery-plugin.html
|
||||||
|
* Timeago - http://timeago.yarp.com/
|
||||||
|
|
9
protected/docs/guide/administration/console.md
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
Console
|
||||||
|
=======
|
||||||
|
|
||||||
|
Zamm provides several console commands to manage the application.
|
||||||
|
|
||||||
|
Run ``./protected/yiic help`` for a full overview.
|
||||||
|
|
||||||
|
Make sure /protected/yiic or /protected/yiic.bat are executable.
|
||||||
|
|
4
protected/docs/guide/administration/cronjobs.md
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
CronJobs
|
||||||
|
========
|
||||||
|
|
||||||
|
See ``Administration -> Settings -> CronJobs`` for a complete list of required cronjobs.
|
23
protected/docs/guide/administration/index.md
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
Administration Guide/Notes
|
||||||
|
===========================
|
||||||
|
|
||||||
|
|
||||||
|
##Installation
|
||||||
|
- [Requirements](requirements.md)
|
||||||
|
- [Installer](installer.md)
|
||||||
|
- [Cronjobs](cronjobs.md)
|
||||||
|
|
||||||
|
##Updating
|
||||||
|
- [Updating](updating.md)
|
||||||
|
|
||||||
|
|
||||||
|
##Advanced Topics
|
||||||
|
- [Logging](logging.md)
|
||||||
|
- [Security](security.md)
|
||||||
|
- [Console](console.md)
|
||||||
|
- [Search](search.md)
|
||||||
|
- [X-Sendfile](xsendfile.md)
|
||||||
|
- [LDAP](ldap.md)
|
||||||
|
|
||||||
|
## More
|
||||||
|
- [Bundled Software](bundled_software.md)
|
8
protected/docs/guide/administration/installer.md
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
Installation using the installer
|
||||||
|
=================================
|
||||||
|
|
||||||
|
1. Create a mysql database
|
||||||
|
2. Extract install package in a web/htdocs folder
|
||||||
|
3. Open that folder (e.g. http://localhost/humhub)
|
||||||
|
4. Installer will start
|
||||||
|
|
47
protected/docs/guide/administration/ldap.md
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
LDAP Support
|
||||||
|
============
|
||||||
|
|
||||||
|
This feature is highly experimental and requires PHP LDAP Support!
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Configuration
|
||||||
|
--------------
|
||||||
|
|
||||||
|
Add following under ``protected/config/_settings.php`` in the ´´params -> auth`` array.
|
||||||
|
|
||||||
|
// LDAP Specifc Settings
|
||||||
|
'ldap' => array(
|
||||||
|
|
||||||
|
// User Group
|
||||||
|
'userGroup' => 'CN=office,CN=Users,DC=domain,DC=local',
|
||||||
|
|
||||||
|
// Account
|
||||||
|
'adminUser' => 'Admin User',
|
||||||
|
'adminPassword' => 'xxx',
|
||||||
|
|
||||||
|
// Connection String
|
||||||
|
'adldapConfig' => array('base_dn'=>'DC=domain,DC=local','account_suffix'=>'@domain.local', 'domain_controllers'=>array('localhost')),
|
||||||
|
|
||||||
|
// Mapping between LDAP Group and Space
|
||||||
|
'groupToSpaceMap' => array(
|
||||||
|
'CN=someGroup,CN=Users,DC=domain,DC=local' => 'Office',
|
||||||
|
// ...
|
||||||
|
),
|
||||||
|
|
||||||
|
// Field Sync (LDAP -> User Field)
|
||||||
|
'fieldMapping' => array(
|
||||||
|
'lastname' => array('ldapField'=>'sn', 'sync'=>'fromLdap'),
|
||||||
|
'firstname' => array('ldapField'=>'givenname', 'sync'=>'fromLdap'),
|
||||||
|
//'city' => array('ldapField'=>'givenname', 'sync'=>'full'),
|
||||||
|
//'state' => array('ldapField'=>'st', 'sync'=>'full'),
|
||||||
|
//'country' => array('ldapField'=>'c', 'sync'=>'fromLdap'),
|
||||||
|
//'country' => array('ldapField'=>'c', 'sync'=>'fromLdap'),
|
||||||
|
),
|
||||||
|
|
||||||
|
),
|
||||||
|
|
||||||
|
LDAP also needs a cronjob
|
||||||
|
|
||||||
|
# When using LDAP
|
||||||
|
45 * * * * /path/to/protected/protected/yiic ldap_update >/dev/null 2>&1
|
4
protected/docs/guide/administration/logging.md
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
Logging
|
||||||
|
============
|
||||||
|
|
||||||
|
Log Files are stored unter ``protected/runtime/application.log``
|
21
protected/docs/guide/administration/requirements.md
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
Requirements
|
||||||
|
============
|
||||||
|
|
||||||
|
### General Requirements:
|
||||||
|
* PHP 5.2+
|
||||||
|
* MySQL
|
||||||
|
* Apache 2.x
|
||||||
|
* ImageMagick
|
||||||
|
* CUrl PHP Extension (For OEmbed Features)
|
||||||
|
|
||||||
|
|
||||||
|
### Packages for Debian/Ubuntu users:
|
||||||
|
* imagemagick
|
||||||
|
* php5-curl
|
||||||
|
* php5-mysql
|
||||||
|
* php5-gd
|
||||||
|
* php5-cli
|
||||||
|
* php5-ldap (optional)
|
||||||
|
* php-apc (optional)
|
||||||
|
* php5-memcached (optional)
|
||||||
|
* libapache2-mod-xsendfile (optional)
|