--- /dev/null
+*.o
+.deps
+aclocal.m4
+autom4te.cache
+config.*
+configure
+depcomp
+install-sh
+Makefile
+Makefile.in
+missing
+stamp-h1
+
+midori
--- /dev/null
+Written by:
+ Christian Dywan <christian@twotoasts.de>
+
+Artwork by:
+ Nancy Runge <nancy@twotoasts.de>
+
+Translations:
+ de: Christian Dywan <christian@twotoasts.de>
\ No newline at end of file
--- /dev/null
+ GNU LESSER GENERAL PUBLIC LICENSE
+ Version 2.1, February 1999
+
+ Copyright (C) 1991, 1999 Free Software Foundation, Inc.
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+[This is the first released version of the Lesser GPL. It also counts
+ as the successor of the GNU Library Public License, version 2, hence
+ the version number 2.1.]
+
+ Preamble
+
+ The licenses for most software are designed to take away your
+freedom to share and change it. By contrast, the GNU General Public
+Licenses are intended to guarantee your freedom to share and change
+free software--to make sure the software is free for all its users.
+
+ This license, the Lesser General Public License, applies to some
+specially designated software packages--typically libraries--of the
+Free Software Foundation and other authors who decide to use it. You
+can use it too, but we suggest you first think carefully about whether
+this license or the ordinary General Public License is the better
+strategy to use in any particular case, based on the explanations below.
+
+ When we speak of free software, we are referring to freedom of use,
+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 this service if you wish); that you receive source code or can get
+it if you want it; that you can change the software and use pieces of
+it in new free programs; and that you are informed that you can do
+these things.
+
+ To protect your rights, we need to make restrictions that forbid
+distributors to deny you these rights or to ask you to surrender these
+rights. These restrictions translate to certain responsibilities for
+you if you distribute copies of the library or if you modify it.
+
+ For example, if you distribute copies of the library, whether gratis
+or for a fee, you must give the recipients all the rights that we gave
+you. You must make sure that they, too, receive or can get the source
+code. If you link other code with the library, you must provide
+complete object files to the recipients, so that they can relink them
+with the library after making changes to the library and recompiling
+it. And you must show them these terms so they know their rights.
+
+ We protect your rights with a two-step method: (1) we copyright the
+library, and (2) we offer you this license, which gives you legal
+permission to copy, distribute and/or modify the library.
+
+ To protect each distributor, we want to make it very clear that
+there is no warranty for the free library. Also, if the library is
+modified by someone else and passed on, the recipients should know
+that what they have is not the original version, so that the original
+author's reputation will not be affected by problems that might be
+introduced by others.
+\f
+ Finally, software patents pose a constant threat to the existence of
+any free program. We wish to make sure that a company cannot
+effectively restrict the users of a free program by obtaining a
+restrictive license from a patent holder. Therefore, we insist that
+any patent license obtained for a version of the library must be
+consistent with the full freedom of use specified in this license.
+
+ Most GNU software, including some libraries, is covered by the
+ordinary GNU General Public License. This license, the GNU Lesser
+General Public License, applies to certain designated libraries, and
+is quite different from the ordinary General Public License. We use
+this license for certain libraries in order to permit linking those
+libraries into non-free programs.
+
+ When a program is linked with a library, whether statically or using
+a shared library, the combination of the two is legally speaking a
+combined work, a derivative of the original library. The ordinary
+General Public License therefore permits such linking only if the
+entire combination fits its criteria of freedom. The Lesser General
+Public License permits more lax criteria for linking other code with
+the library.
+
+ We call this license the "Lesser" General Public License because it
+does Less to protect the user's freedom than the ordinary General
+Public License. It also provides other free software developers Less
+of an advantage over competing non-free programs. These disadvantages
+are the reason we use the ordinary General Public License for many
+libraries. However, the Lesser license provides advantages in certain
+special circumstances.
+
+ For example, on rare occasions, there may be a special need to
+encourage the widest possible use of a certain library, so that it becomes
+a de-facto standard. To achieve this, non-free programs must be
+allowed to use the library. A more frequent case is that a free
+library does the same job as widely used non-free libraries. In this
+case, there is little to gain by limiting the free library to free
+software only, so we use the Lesser General Public License.
+
+ In other cases, permission to use a particular library in non-free
+programs enables a greater number of people to use a large body of
+free software. For example, permission to use the GNU C Library in
+non-free programs enables many more people to use the whole GNU
+operating system, as well as its variant, the GNU/Linux operating
+system.
+
+ Although the Lesser General Public License is Less protective of the
+users' freedom, it does ensure that the user of a program that is
+linked with the Library has the freedom and the wherewithal to run
+that program using a modified version of the Library.
+
+ The precise terms and conditions for copying, distribution and
+modification follow. Pay close attention to the difference between a
+"work based on the library" and a "work that uses the library". The
+former contains code derived from the library, whereas the latter must
+be combined with the library in order to run.
+\f
+ GNU LESSER GENERAL PUBLIC LICENSE
+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+ 0. This License Agreement applies to any software library or other
+program which contains a notice placed by the copyright holder or
+other authorized party saying it may be distributed under the terms of
+this Lesser General Public License (also called "this License").
+Each licensee is addressed as "you".
+
+ A "library" means a collection of software functions and/or data
+prepared so as to be conveniently linked with application programs
+(which use some of those functions and data) to form executables.
+
+ The "Library", below, refers to any such software library or work
+which has been distributed under these terms. A "work based on the
+Library" means either the Library or any derivative work under
+copyright law: that is to say, a work containing the Library or a
+portion of it, either verbatim or with modifications and/or translated
+straightforwardly into another language. (Hereinafter, translation is
+included without limitation in the term "modification".)
+
+ "Source code" for a work means the preferred form of the work for
+making modifications to it. For a library, complete source code means
+all the source code for all modules it contains, plus any associated
+interface definition files, plus the scripts used to control compilation
+and installation of the library.
+
+ Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope. The act of
+running a program using the Library is not restricted, and output from
+such a program is covered only if its contents constitute a work based
+on the Library (independent of the use of the Library in a tool for
+writing it). Whether that is true depends on what the Library does
+and what the program that uses the Library does.
+
+ 1. You may copy and distribute verbatim copies of the Library's
+complete source code as you receive it, in any medium, provided that
+you conspicuously and appropriately publish on each copy an
+appropriate copyright notice and disclaimer of warranty; keep intact
+all the notices that refer to this License and to the absence of any
+warranty; and distribute a copy of this License along with the
+Library.
+
+ You may charge a fee for the physical act of transferring a copy,
+and you may at your option offer warranty protection in exchange for a
+fee.
+\f
+ 2. You may modify your copy or copies of the Library or any portion
+of it, thus forming a work based on the Library, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+ a) The modified work must itself be a software library.
+
+ b) You must cause the files modified to carry prominent notices
+ stating that you changed the files and the date of any change.
+
+ c) You must cause the whole of the work to be licensed at no
+ charge to all third parties under the terms of this License.
+
+ d) If a facility in the modified Library refers to a function or a
+ table of data to be supplied by an application program that uses
+ the facility, other than as an argument passed when the facility
+ is invoked, then you must make a good faith effort to ensure that,
+ in the event an application does not supply such function or
+ table, the facility still operates, and performs whatever part of
+ its purpose remains meaningful.
+
+ (For example, a function in a library to compute square roots has
+ a purpose that is entirely well-defined independent of the
+ application. Therefore, Subsection 2d requires that any
+ application-supplied function or table used by this function must
+ be optional: if the application does not supply it, the square
+ root function must still compute square roots.)
+
+These requirements apply to the modified work as a whole. If
+identifiable sections of that work are not derived from the Library,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works. But when you
+distribute the same sections as part of a whole which is a work based
+on the Library, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote
+it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Library.
+
+In addition, mere aggregation of another work not based on the Library
+with the Library (or with a work based on the Library) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+ 3. You may opt to apply the terms of the ordinary GNU General Public
+License instead of this License to a given copy of the Library. To do
+this, you must alter all the notices that refer to this License, so
+that they refer to the ordinary GNU General Public License, version 2,
+instead of to this License. (If a newer version than version 2 of the
+ordinary GNU General Public License has appeared, then you can specify
+that version instead if you wish.) Do not make any other change in
+these notices.
+\f
+ Once this change is made in a given copy, it is irreversible for
+that copy, so the ordinary GNU General Public License applies to all
+subsequent copies and derivative works made from that copy.
+
+ This option is useful when you wish to copy part of the code of
+the Library into a program that is not a library.
+
+ 4. You may copy and distribute the Library (or a portion or
+derivative of it, under Section 2) in object code or executable form
+under the terms of Sections 1 and 2 above provided that you accompany
+it with the complete corresponding machine-readable source code, which
+must be distributed under the terms of Sections 1 and 2 above on a
+medium customarily used for software interchange.
+
+ If distribution of object code is made by offering access to copy
+from a designated place, then offering equivalent access to copy the
+source code from the same place satisfies the requirement to
+distribute the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+ 5. A program that contains no derivative of any portion of the
+Library, but is designed to work with the Library by being compiled or
+linked with it, is called a "work that uses the Library". Such a
+work, in isolation, is not a derivative work of the Library, and
+therefore falls outside the scope of this License.
+
+ However, linking a "work that uses the Library" with the Library
+creates an executable that is a derivative of the Library (because it
+contains portions of the Library), rather than a "work that uses the
+library". The executable is therefore covered by this License.
+Section 6 states terms for distribution of such executables.
+
+ When a "work that uses the Library" uses material from a header file
+that is part of the Library, the object code for the work may be a
+derivative work of the Library even though the source code is not.
+Whether this is true is especially significant if the work can be
+linked without the Library, or if the work is itself a library. The
+threshold for this to be true is not precisely defined by law.
+
+ If such an object file uses only numerical parameters, data
+structure layouts and accessors, and small macros and small inline
+functions (ten lines or less in length), then the use of the object
+file is unrestricted, regardless of whether it is legally a derivative
+work. (Executables containing this object code plus portions of the
+Library will still fall under Section 6.)
+
+ Otherwise, if the work is a derivative of the Library, you may
+distribute the object code for the work under the terms of Section 6.
+Any executables containing that work also fall under Section 6,
+whether or not they are linked directly with the Library itself.
+\f
+ 6. As an exception to the Sections above, you may also combine or
+link a "work that uses the Library" with the Library to produce a
+work containing portions of the Library, and distribute that work
+under terms of your choice, provided that the terms permit
+modification of the work for the customer's own use and reverse
+engineering for debugging such modifications.
+
+ You must give prominent notice with each copy of the work that the
+Library is used in it and that the Library and its use are covered by
+this License. You must supply a copy of this License. If the work
+during execution displays copyright notices, you must include the
+copyright notice for the Library among them, as well as a reference
+directing the user to the copy of this License. Also, you must do one
+of these things:
+
+ a) Accompany the work with the complete corresponding
+ machine-readable source code for the Library including whatever
+ changes were used in the work (which must be distributed under
+ Sections 1 and 2 above); and, if the work is an executable linked
+ with the Library, with the complete machine-readable "work that
+ uses the Library", as object code and/or source code, so that the
+ user can modify the Library and then relink to produce a modified
+ executable containing the modified Library. (It is understood
+ that the user who changes the contents of definitions files in the
+ Library will not necessarily be able to recompile the application
+ to use the modified definitions.)
+
+ b) Use a suitable shared library mechanism for linking with the
+ Library. A suitable mechanism is one that (1) uses at run time a
+ copy of the library already present on the user's computer system,
+ rather than copying library functions into the executable, and (2)
+ will operate properly with a modified version of the library, if
+ the user installs one, as long as the modified version is
+ interface-compatible with the version that the work was made with.
+
+ c) Accompany the work with a written offer, valid for at
+ least three years, to give the same user the materials
+ specified in Subsection 6a, above, for a charge no more
+ than the cost of performing this distribution.
+
+ d) If distribution of the work is made by offering access to copy
+ from a designated place, offer equivalent access to copy the above
+ specified materials from the same place.
+
+ e) Verify that the user has already received a copy of these
+ materials or that you have already sent this user a copy.
+
+ For an executable, the required form of the "work that uses the
+Library" must include any data and utility programs needed for
+reproducing the executable from it. However, as a special exception,
+the materials to be distributed need not include anything that is
+normally distributed (in either source or binary form) with the major
+components (compiler, kernel, and so on) of the operating system on
+which the executable runs, unless that component itself accompanies
+the executable.
+
+ It may happen that this requirement contradicts the license
+restrictions of other proprietary libraries that do not normally
+accompany the operating system. Such a contradiction means you cannot
+use both them and the Library together in an executable that you
+distribute.
+\f
+ 7. You may place library facilities that are a work based on the
+Library side-by-side in a single library together with other library
+facilities not covered by this License, and distribute such a combined
+library, provided that the separate distribution of the work based on
+the Library and of the other library facilities is otherwise
+permitted, and provided that you do these two things:
+
+ a) Accompany the combined library with a copy of the same work
+ based on the Library, uncombined with any other library
+ facilities. This must be distributed under the terms of the
+ Sections above.
+
+ b) Give prominent notice with the combined library of the fact
+ that part of it is a work based on the Library, and explaining
+ where to find the accompanying uncombined form of the same work.
+
+ 8. You may not copy, modify, sublicense, link with, or distribute
+the Library except as expressly provided under this License. Any
+attempt otherwise to copy, modify, sublicense, link with, or
+distribute the Library is void, and will automatically terminate your
+rights under this License. However, parties who have received copies,
+or rights, from you under this License will not have their licenses
+terminated so long as such parties remain in full compliance.
+
+ 9. You are not required to accept this License, since you have not
+signed it. However, nothing else grants you permission to modify or
+distribute the Library or its derivative works. These actions are
+prohibited by law if you do not accept this License. Therefore, by
+modifying or distributing the Library (or any work based on the
+Library), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Library or works based on it.
+
+ 10. Each time you redistribute the Library (or any work based on the
+Library), the recipient automatically receives a license from the
+original licensor to copy, distribute, link with or modify the Library
+subject to these terms and conditions. You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties with
+this License.
+\f
+ 11. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+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
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Library at all. For example, if a patent
+license would not permit royalty-free redistribution of the Library by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Library.
+
+If any portion of this section is held invalid or unenforceable under any
+particular circumstance, the balance of the section is intended to apply,
+and the section as a whole is intended to apply in other circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system which is
+implemented by public license practices. Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+ 12. If the distribution and/or use of the Library is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Library under this License may add
+an explicit geographical distribution limitation excluding those countries,
+so that distribution is permitted only in or among countries not thus
+excluded. In such case, this License incorporates the limitation as if
+written in the body of this License.
+
+ 13. The Free Software Foundation may publish revised and/or new
+versions of the Lesser 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 Library
+specifies a version number of this License which applies to it and
+"any later version", you have the option of following the terms and
+conditions either of that version or of any later version published by
+the Free Software Foundation. If the Library does not specify a
+license version number, you may choose any version ever published by
+the Free Software Foundation.
+\f
+ 14. If you wish to incorporate parts of the Library into other free
+programs whose distribution conditions are incompatible with these,
+write to the author to ask for permission. For software which is
+copyrighted by the Free Software Foundation, write to the Free
+Software Foundation; we sometimes make exceptions for this. Our
+decision will be guided by the two goals of preserving the free status
+of all derivatives of our free software and of promoting the sharing
+and reuse of software generally.
+
+ NO WARRANTY
+
+ 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO
+WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW.
+EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR
+OTHER PARTIES PROVIDE THE LIBRARY "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
+LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME
+THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+ 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN
+WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY
+AND/OR REDISTRIBUTE THE LIBRARY 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
+LIBRARY (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 LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF
+SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
+DAMAGES.
+
+ END OF TERMS AND CONDITIONS
+\f
+ How to Apply These Terms to Your New Libraries
+
+ If you develop a new library, and you want it to be of the greatest
+possible use to the public, we recommend making it free software that
+everyone can redistribute and change. You can do so by permitting
+redistribution under these terms (or, alternatively, under the terms of the
+ordinary General Public License).
+
+ To apply these terms, attach the following notices to the library. It is
+safest to attach them to the start of each source file to most effectively
+convey 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 library's name and a brief idea of what it does.>
+ Copyright (C) <year> <name of author>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ This library 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
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with this library; if not, write to the Free Software
+ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+
+Also add information on how to contact you by electronic and paper mail.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the library, if
+necessary. Here is a sample; alter the names:
+
+ Yoyodyne, Inc., hereby disclaims all copyright interest in the
+ library `Frob' (a library for tweaking knobs) written by James Random Hacker.
+
+ <signature of Ty Coon>, 1 April 1990
+ Ty Coon, President of Vice
+
+That's all there is to it!
+
+
--- /dev/null
+v0.0.14:
+ + FIX Reopening a tab from the trash causes a crash
+ + FIX An untitled website keeps the previous title
+ + FIX When switching tabs the location/ title isn't updated correctly
+ + FIX Issues with the preferences dialog
+ + Disable location completion for now
+ + Save tabtrash to file
+ + Restructure some code
+ + Remove color picker and throbber
+ + Change the license to LGPL
+
+v0.0.13:
+ + Adapt WebKit api change, remove engine wrappers, remove dialog hack
+ + Improve XBEL loading and saving
+ + Show dialog and backup files on startup errors
+ + Rearranged and removed some menus
+
+v0.0.12:
+ + FIX Improve flawed window creation
+ + Build with and eliminate all compiler warnings
+ + Implement clipboard handling menus
+ + Allow editing of search engines
+ + Update search engines properly
+ + Implement bookmarks saving
+ + Implement session saving and loading
+ + Cleaned up and revised most code
+ + Remove legacy webi code
+
+v0.0.11:
+ + FIX Back/ forward and initial check menu item states
+ + Remove rather useless debugging helpers
+ + Improvements on the preferences
+ + First attempt at websearch
+ + Preserve typed uri on tab switch
+ + First attempt at bookmarks, readonly for now
+ + Add an animated throbber
+
+v0.0.10:
+ + FIX Can crash on context menu or new protocol
+ + FIX Location isn't updated on tab switch
+ + Remember last window position and size
+ + Implement Open menu item
+ + Allow using location and web search if hidden
+
+v0.0.9:
+ + FIX Close tab not insensitive for only one tab
+ + FIX Debug output is broken
+ + Display uri when hovering a link
+ + Implement link uri related part of context menu
+ + Implement alt/ middle/ shift click link
+ + First attempt at external protocol handlers
+ + Initial download manager integration
+ + Adapt WebKit api prefix change
+ + More code reorganization and cleanup
+
+v0.0.8:
+ + FIX Crash when invoking document context menu via keyboard
+ + FIX Can't build with debug = yes on GTK+2.12
+ + Changes related to icons in the gui
+ + Reorganize code by splitting into several files
+ + Switch from WebkitGdk to WebkitGtk
+
+v0.0.7:
+ + FIX Make settings finally work flawlessly
+ + FIX Can crash when settings are opened
+ + Handle all panels in a general way
+ + Install xdg compliant desktop file
+ + Implement location and web search menu items
+ + Display a loading icon on tabs again
+ + Changed the settings dialog again
+
+v0.0.6:
+ + FIX Closing an individual tab doesn't work correctly.
+ + FIX Doesn't build with gtkwebcore.
+ + Reimplement menus and and navibar with GtkUIManager.
+ + Improve document handling in general.
+ + Finished tab trash menu.
+ + Implement searchbox default text.
+ + Remove some gtkwebcore code.
+ + Use Xfce style dialog in Xfce.
+ + Implement a few settings.
+ + Change the panel's look.
+ + Implement a 'pageholder' panel.
+
+v0.0.5:
+ + Implement a few more signals for WebkitGdk.
+ + Add tooltips to navigation toolbar buttons.
+ + First attempt on a settings dialog.
+ + Reimplemented color picker.
+ + Autocompletion for location and searchbox.
+ + Changed menu items and incremental findbar.
+ + Implement tab switching via keyboard.
+
+v0.0.4:
+ + FIX Midori segfaults when quitting.
+ + FIX Config loading and saving is broken.
+ + Switch WebkitGdk to gtk api and make it the build default.
+ + Register custom stock icons instead of icon theme magic.
+ + Implement dynamic window menu.
+ + First attempt on resizable panels.
+ + Add about dialog.
+
+v0.0.3:
+ + FIX Refresh via menu or shortcut crashes the browser.
+ + FIX Assertions with and visibility of the progressbar.
+ + FIX Tabs are not reorderable.
+ + Package versions in ./configure result and --version output.
+ + Improve view menu and add tools menu.
+ + Replace deprecated functions and macros.
+ + Implement settings saving and loading.
+ + Fill the common context menu with items
+ + Allow multiple homepages, seperated by '|'.
+ + Make code typesafe and C++ friendly.
+ + Initially support WebkitGdk directly.
+ + Urlify location inputs automatically.
+
+v0.0.2:
+ + Dynamic tab trash menu.
+ + Update UI when page is changed.
+ + Enhanced WebkitGtk support.
+ + New function sokoke_dialog_run_modeless.
+ + Finished on_document_request_script_prompt.
+ + One name and version, visible in the user agent.
+ + Changed some accelerators and menu items.
+ + Create and destroy color picker properly.
+ + Ctrl + Wheel resets the zoom level.
+ + Escape in the location entry resets the uri.
+ + Use gtk-webcore prefix instead of osb now.
+ + Save keybindings on quit.
+ + Fancy autotools build setup.
+
+v0.0.1:
+ + Initial release
--- /dev/null
+It is 4 spaces, no tabs, preferrably at 80 columns per line.
+
+The preferred coding style is explained by example.
+
+Source file example:
+
+ /*
+ Copyright
+ LICENSE TEXT
+ */
+
+ #include "foo.h"
+
+ #include "bar.h"
+
+ #include <glib.h>
+
+ void foobar(FooEnum bar, const gchar* foo)
+ {
+ if(!foo)
+ return;
+
+ #ifdef BAR_STRICT
+ if(bar == FOO_N)
+ {
+ g_print("illegal value for 'bar'.\n");
+ return;
+ }
+ #endif
+
+ // this is an example
+ gint n;
+ switch(bar)
+ {
+ case FOO_FOO:
+ n = bar + 1;
+ break;
+ case FOO_BAR:
+ n = bar + 10;
+ break;
+ default:
+ n = 1;
+ }
+
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ g_print("%s\n", foo);
+ }
+ }
+
+Header file example:
+
+ /*
+ Copyright
+ LICENSE TEXT
+ */
+
+ #ifndef __FOO_H__
+ #define __FOO_H__ 1
+
+ #ifdef HAVE_BAR_H
+ #define BAR_STRICT
+ #endif
+
+ // -- Types
+
+ typedef enum
+ {
+ FOO_FOO,
+ FOO_BAR,
+ FOO_N
+ } FooEnum;
+
+ typedef struct
+ {
+ FooEnum fooBar;
+ } FooStruct;
+
+ // -- Declarations
+
+ void
+ foobar(FooEnum, const gchar*);
+
+ #endif /* !__FOO_H__ */
--- /dev/null
+1. Unpack the archive.
+
+2. Check build options with
+ ./configure --help
+
+3. Prepare for build with
+ ./configure
+
+4. Build midori with
+ make
+
+5. To install midori run either
+ (sudo) make install
+ or
+ (sudo) checkinstall
+
+6. For a list of command line options run
+ midori --help
\ No newline at end of file
--- /dev/null
+AUTOMAKE_OPTIONS = gnu
+
+SUBDIRS = src
+
+desktopdir = $(datadir)/applications
+desktop_DATA = midori.desktop
+
+EXTRA_DIST = HACKING TODO $(desktop_DATA)
--- /dev/null
+Midori is a lightweight web browser.
+
+* Full integration with GTK+2.
+* Fast rendering with WebKit.
+* Tabs, windows and session management.
+* Bookmarks are stored with XBEL.
+* Searchbox based on OpenSearch.
+* Custom context menu actions.
+* User scripts and user styles support.
+* Extensible via Lua scripts.
+
+Requirements: GTK+ 2.6, libsexy, WebkitGtk, libXML2
+
+For installation instructions read INSTALL.
+
+Please report comments, suggestions and bugs to:
+ Christian Dywan <christian@twotoasts.de>
+
+Check for new versions at:
+ http://software.twotoasts.de
--- /dev/null
+TODO:
+ . Save files on change as opposed to on quit
+ . Position menus properly
+ . Use an animated throbber
+ . Add a context menu to the pageholder
+ . Make tabs autoshrink
+ . New tabs from a TabNew action should always open in the foreground
+ . Middle click toolbuttons or menuitems should open new tabs
+ . Implement userscript support
+ . Actual multiple window support, some things just ignore this currently
+ . Custom context menu actions
+ . custom tab names
+ . Open an auto-vanishing findbox with '.'
+ . Support gettext
+ . Custom panels, loaded from (x)htm(l) files or websites
+ . Drag tabs onto the panel to have it in the sidebar.
+ . Save completion stores
+ . analogus to blocked popups, blocked scripts moving layers on load
+ . per-site blocking of individual elements on a page
+ . statusbar icon 'cookies blocked', icon 'popups blocked'
+ . per-site settings accessible via statusbar icons, ie. cookies, popups, plugins
+ . cookieSafe like, a list of cookies, with type, block, allow
+ . support mouse gestures
+ . optional internal source view using gtksourceview
+ . automatic update checks (browser, extensions)?
+ . auto-group tabs by opener, with colors?
+ . mark (dogear) a selection so that it isn't cleared implicitly, multiply on one page
+ . have an internal uri scheme, eg. 'res:', to reference eg. themed icons
+ . 'about:' uris: about, blank, cache, config, plugins
+ . panel of open tabs (with tree-structure), optional thumbnail-view
+ . spell check support
+ . allow full page zoom (how do we incorporate it in the gui?)
+ . check specific bookmarks for updates automatically (as an extension?)
+ . mark "new" as well as "actually modified" tabs specially (even over sessions)
+ . customizable toolbars, custom buttons (uri, title, icon)
+ . searchEngine: "Show in context menu"
+ . use libnotify for events, e.g. download finished
+ . save screenshot of a document
+ . right-click a textbox in a search form and choose 'add to websearch'
+ . support extensions written in lua
+ . the scrollbar must be exactly at the right (left) edge if maximized: fitt's law
+ . detailed css element view, maybe in 'properties'?
+ . reuse running instance, probably via libunique
+ . respect design principle "no warnings but undo of backups"?
+ . support widgets 1.0 spec in tool windows and standalone
+ . blank page: several custom links, displayed as thumbnails, like Opera
+ . handle downloads, optionally in a downloadbar
+ . Implement userstyle support
+ . Protected tabs prompt when attempting to close them
+ . provide a 'sleep mode' after a crash where open documents are loaded manually
+ . option to run plugins or scripts only on demand, like NoScript, per-site
+ . optional http redirection manually or on timeout
+ . style: none, compatible (b/w), default, [styles], "media", ["media" styles], ...
+ . mouse pointer coordinates in the status bar
+ . draw rectangle with the mouse, x/y/x2/y2 in the statusbar
+ . formfill (like Opera's magic wand)
+ . private browsing mode (no browsing, download or search history)
+ . shared bookmarks and config
+ . custom-mode, e.g. hide menubar and use help icon to have a help viewer
+ . dead tabs: download, aborted page
+ . on url load, for big files, ask "Open or save?"
+ . middle-click on selection to open <selection>
--- /dev/null
+#!/bin/sh
+
+aclocal
+autoheader
+autoconf
+automake --add-missing --copy
\ No newline at end of file
--- /dev/null
+# Register ourselves to autoconf
+AC_INIT([midori], [0.0.14], [christian@twotoasts.de])
+AC_CONFIG_SRCDIR([src/main.h])
+AC_CONFIG_HEADER([config.h])
+
+AM_INIT_AUTOMAKE([AC_PACKAGE_TARNAME()], [AC_PACKAGE_VERSION()])
+
+# Checks for programs
+AC_PROG_CC
+AC_PROG_INSTALL
+AC_PROG_MAKE_SET
+
+# Checks for header files
+AC_HEADER_STDC
+
+# Checks for typedefs, structures, and compiler characteristics.
+AC_C_CONST
+
+# Checks if we want debugging support
+AC_ARG_ENABLE([debug],
+AC_HELP_STRING([--enable-debug=@<:@no/simple/yes@:>@]
+ , [Turn on debugging @<:@default=simple@:>@])
+ , [], [enable_debug=simple])
+AC_MSG_CHECKING([whether to enable debugging support])
+AC_MSG_RESULT([$enable_debug])
+if test x"$enable_debug" = x"simple"; then
+ AC_DEFINE([SOKOKE_DEBUG], 1, [Level of debugging support])
+fi
+if test x"$enable_debug" = x"yes"; then
+ # Check whether the compiler accepts -Wall
+ save_CFLAGS="$CFLAGS"
+ CFLAGS="$CFLAGS -Wall"
+ AC_MSG_CHECKING([whether $CC accepts -Wall])
+ AC_COMPILE_IFELSE(AC_LANG_SOURCE([int x;]), [
+ AC_MSG_RESULT([yes])
+ ], [
+ AC_MSG_RESULT([no])
+ CFLAGS="$save_CFLAGS"
+ ])
+ AC_DEFINE([SOKOKE_DEBUG], 2, [Level of debugging support])
+fi
+AC_DEFINE_UNQUOTED([SOKOKE_DEBUG_], "$enable_debug", [Debugging?])
+
+# Checks for GTK+2
+PKG_CHECK_MODULES(GTK, gtk+-2.0 >= 2.6, have_gtk=true, have_gtk=false)
+if test "x${have_gtk}" = "xfalse" ; then
+ AC_MSG_ERROR([No GTK+2 package information found])
+fi
+AC_SUBST(GTK_CFLAGS)
+AC_SUBST(GTK_LIBS)
+GTK_VER=`pkg-config --modversion gtk+-2.0`
+AC_DEFINE_UNQUOTED([GTK_VER], "$GTK_VER", [GTK+ version])
+
+# Checks for WebKitGtk
+PKG_CHECK_MODULES(WEBKIT, WebKitGtk, have_webkit=true, have_webkit=false)
+if test "x${have_webkit}" = "xfalse" ; then
+ AC_MSG_ERROR([No WebKitGtk package information found])
+fi
+AC_SUBST(WEBKIT_CFLAGS)
+AC_SUBST(WEBKIT_LIBS)
+WEBKIT_VER=`pkg-config --modversion WebKitGtk`
+AC_DEFINE_UNQUOTED([WEBKIT_VER], "$WEBKITGTK_VER", [WebKitGtk version])
+
+# Checks for libsexy
+PKG_CHECK_MODULES(LIBSEXY, libsexy, have_libsexy=true, have_libsexy=false)
+if test "x${have_libsexy}" = "xfalse" ; then
+ AC_MSG_ERROR([No Libsexy package information found])
+fi
+AC_SUBST(LIBSEXY_CFLAGS)
+AC_SUBST(LIBSEXY_LIBS)
+LIBSEXY_VER=`pkg-config --modversion libsexy`
+AC_DEFINE_UNQUOTED([LIBSEXY_VER], "$LIBSEXY_VER", [Libsexy version])
+
+# Checks for LibXML2
+PKG_CHECK_MODULES(LIBXML, libxml-2.0 >= 2.6, have_libxml=true, have_libxml=false)
+if test "x${have_libxml}" = "xfalse" ; then
+ AC_MSG_ERROR([No libXML2 package information found])
+fi
+AC_SUBST(LIBXML_CFLAGS)
+AC_SUBST(LIBXML_LIBS)
+LIBXML_VER=`pkg-config --modversion libxml-2.0`
+AC_DEFINE_UNQUOTED([LIBXML_VER], "$LIBXML_VER", [libXML2 version])
+
+# Here we tell the configure script which files to *create*
+AC_CONFIG_FILES([
+ Makefile \
+ src/Makefile
+])
+AC_OUTPUT
+
+# Show us what we have
+echo
+echo " GTK+2 $GTK_VER"
+echo " WebKit $WEBKIT_VER"
+echo " Libsexy $LIBSEXY_VER"
+echo " libXML2 $LIBXML_VER"
+echo " GetText N/A"
+echo
+echo " Debugging $enable_debug"
+echo
+echo " Prefix $prefix"
--- /dev/null
+
+[Desktop Entry]
+Version=1.0
+Encoding=UTF-8
+Type=Application
+Name=Midori Web Browser
+GenericName=Web Browser
+Comment=Lightweight web browser
+Categories=GTK;Network;
+MimeType=text/html;text/xml;application/xhtml+xml;application/xml
+Exec=midori %u
+Icon=web-browser
+Terminal=false
+StartupNotify=true
--- /dev/null
+INCLUDES = $(GTK_CFLAGS) $(WEBKIT_CFLAGS) $(LIBXML_CFLAGS) $(LIBSEXY_CFLAGS)
+LDADD = $(GTK_LIBS) $(WEBKIT_LIBS) $(LIBXML_LIBS) $(LIBSEXY_LIBS)
+
+bin_PROGRAMS = midori
+midori_SOURCES = main.c main.h \
+ browser.c browser.h \
+ prefs.c prefs.h \
+ webSearch.c webSearch.h \
+ helpers.c helpers.h \
+ webView.c webView.h \
+ sokoke.c sokoke.h \
+ conf.c conf.h \
+ search.c search.h \
+ xbel.c xbel.h \
+ global.h \
+ ui.h \
+ debug.h
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "browser.h"
+
+#include "helpers.h"
+#include "prefs.h"
+#include "sokoke.h"
+#include "ui.h"
+#include "webView.h"
+#include "webSearch.h"
+#include "xbel.h"
+
+#include <gdk/gdkkeysyms.h>
+#include <string.h>
+
+// -- GTK+ signal handlers begin here
+
+void on_action_window_new_activate(GtkAction* action, CBrowser* browser)
+{
+ browser_new(NULL);
+}
+
+void on_action_tab_new_activate(GtkAction* action, CBrowser* browser)
+{
+ browser_new(browser);
+ update_browser_actions(browser);
+}
+
+void on_action_open_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* dialog = gtk_file_chooser_dialog_new("Open file"
+ , GTK_WINDOW(browser->window)
+ , GTK_FILE_CHOOSER_ACTION_OPEN
+ , GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL
+ , GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog), GTK_STOCK_OPEN);
+ if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
+ {
+ gchar* sFilename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
+ webView_open(get_nth_webView(-1, browser), sFilename);
+ g_free(sFilename);
+ }
+ gtk_widget_destroy(dialog);
+}
+
+void on_action_tab_close_activate(GtkAction* action, CBrowser* browser)
+{
+ webView_close(get_nth_webView(-1, browser), browser);
+}
+
+void on_action_window_close_activate(GtkAction* action, CBrowser* browser)
+{
+ gtk_widget_destroy(browser->window);
+}
+
+void on_action_quit_activate(GtkAction* action, CBrowser* browser)
+{
+ gtk_main_quit();
+}
+
+void on_action_edit_activate(GtkAction* action, CBrowser* browser)
+{
+ update_edit_items(browser);
+}
+
+void on_action_cut_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* widget = gtk_window_get_focus(GTK_WINDOW(browser->window));
+ g_signal_emit_by_name(widget, "cut-clipboard");
+}
+
+void on_action_copy_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* widget = gtk_window_get_focus(GTK_WINDOW(browser->window));
+ g_signal_emit_by_name(widget, "copy-clipboard");
+}
+
+void on_action_paste_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* widget = gtk_window_get_focus(GTK_WINDOW(browser->window));
+ g_signal_emit_by_name(widget, "paste-clipboard");
+}
+
+void on_action_delete_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* widget = gtk_window_get_focus(GTK_WINDOW(browser->window));
+ if(WEBKIT_IS_WEB_VIEW(widget))
+ ;//webkit_web_view_delete_selection(WEBKIT_WEB_VIEW(widget));
+ else if(GTK_IS_EDITABLE(widget))
+ gtk_editable_delete_selection(GTK_EDITABLE(widget));
+}
+
+void on_action_selectAll_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* widget = gtk_window_get_focus(GTK_WINDOW(browser->window));
+ if(GTK_IS_ENTRY(widget))
+ gtk_editable_select_region(GTK_EDITABLE(widget), 0, -1);
+ else
+ g_signal_emit_by_name(widget, "select-all");
+}
+
+void on_action_find_activate(GtkAction* action, CBrowser* browser)
+{
+ if(GTK_WIDGET_VISIBLE(browser->findbox))
+ gtk_widget_hide(browser->findbox);
+ else
+ {
+ GtkWidget* icon = gtk_image_new_from_stock(GTK_STOCK_FIND, GTK_ICON_SIZE_MENU);
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->findbox_text)
+ , SEXY_ICON_ENTRY_PRIMARY, GTK_IMAGE(icon));
+ gtk_widget_show(browser->findbox);
+ gtk_widget_grab_focus(GTK_WIDGET(browser->findbox_text));
+ }
+}
+
+void on_action_find_next_activate(GtkAction* action, CBrowser* browser)
+{
+ if(!GTK_WIDGET_VISIBLE(browser->findbox))
+ ; // FIXME: What if the findbox is hidden?
+ const gchar* text = gtk_entry_get_text(GTK_ENTRY(browser->findbox_text));
+ const gboolean caseSensitive = gtk_toggle_tool_button_get_active(
+ GTK_TOGGLE_TOOL_BUTTON(browser->findbox_case));
+ GtkWidget* webView = get_nth_webView(-1, browser);
+ GtkWidget* icon;
+ webkit_web_view_unmark_text_matches(WEBKIT_WEB_VIEW(webView));
+ if(webkit_web_view_search_text(WEBKIT_WEB_VIEW(webView), text, TRUE, caseSensitive, TRUE))
+ icon = gtk_image_new_from_stock(GTK_STOCK_FIND, GTK_ICON_SIZE_MENU);
+ else
+ icon = gtk_image_new_from_stock(GTK_STOCK_STOP, GTK_ICON_SIZE_MENU);
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->findbox_text)
+ , SEXY_ICON_ENTRY_PRIMARY, GTK_IMAGE(icon));
+ webkit_web_view_mark_text_matches(WEBKIT_WEB_VIEW(webView), text, caseSensitive, 0);
+ const gboolean highlight = gtk_toggle_tool_button_get_active(
+ GTK_TOGGLE_TOOL_BUTTON(browser->findbox_highlight));
+ webkit_web_view_set_highlight_text_matches(WEBKIT_WEB_VIEW(webView), highlight);
+}
+
+void on_action_find_previous_activate(GtkAction* action, CBrowser* browser)
+{
+ if(!GTK_WIDGET_VISIBLE(browser->findbox))
+ ; // FIXME: What if the findbox is hidden?
+ const gchar* text = gtk_entry_get_text(GTK_ENTRY(browser->findbox_text));
+ const gboolean caseSensitive = gtk_toggle_tool_button_get_active(
+ GTK_TOGGLE_TOOL_BUTTON(browser->findbox_case));
+ GtkWidget* webView = get_nth_webView(-1, browser);
+ webkit_web_view_unmark_text_matches(WEBKIT_WEB_VIEW(webView));
+ GtkWidget* icon;
+ if(webkit_web_view_search_text(WEBKIT_WEB_VIEW(webView), text, FALSE, caseSensitive, TRUE))
+ icon = gtk_image_new_from_stock(GTK_STOCK_FIND, GTK_ICON_SIZE_MENU);
+ else
+ icon = gtk_image_new_from_stock(GTK_STOCK_STOP, GTK_ICON_SIZE_MENU);
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->findbox_text)
+ , SEXY_ICON_ENTRY_PRIMARY, GTK_IMAGE(icon));
+ webkit_web_view_mark_text_matches(WEBKIT_WEB_VIEW(webView), text, caseSensitive, 0);
+ const gboolean highlight = gtk_toggle_tool_button_get_active(
+ GTK_TOGGLE_TOOL_BUTTON(browser->findbox_highlight));
+ webkit_web_view_set_highlight_text_matches(WEBKIT_WEB_VIEW(webView), highlight);
+}
+
+void on_findbox_highlight_toggled(GtkToggleToolButton* toolitem, CBrowser* browser)
+{
+ GtkWidget* webView = get_nth_webView(-1, browser);
+ const gboolean highlight = gtk_toggle_tool_button_get_active(toolitem);
+ webkit_web_view_set_highlight_text_matches(WEBKIT_WEB_VIEW(webView), highlight);
+}
+
+void on_action_preferences_activate(GtkAction* action, CBrowser* browser)
+{
+ // Show the preferences dialog. Create it if necessary.
+ static GtkWidget* dialog;
+ if(GTK_IS_DIALOG(dialog))
+ gtk_window_present(GTK_WINDOW(dialog));
+ else
+ {
+ dialog = prefs_preferences_dialog_new(browser);
+ gtk_widget_show(dialog);
+ }
+}
+
+static void on_toolbar_navigation_notify_style(GObject* object, GParamSpec* arg1
+ , CBrowser* browser)
+{
+ if(config->toolbarStyle == CONFIG_TOOLBAR_DEFAULT)
+ {
+ gtk_toolbar_set_style(GTK_TOOLBAR(browser->navibar)
+ , config_to_toolbarstyle(config->toolbarStyle));
+ }
+}
+
+void on_action_toolbar_navigation_activate(GtkToggleAction* action, CBrowser* browser)
+{
+ config->toolbarNavigation = gtk_toggle_action_get_active(action);
+ sokoke_widget_set_visible(browser->navibar, config->toolbarNavigation);
+}
+
+void on_action_toolbar_bookmarks_activate(GtkToggleAction* action, CBrowser* browser)
+{
+ config->toolbarBookmarks = gtk_toggle_action_get_active(action);
+ sokoke_widget_set_visible(browser->bookmarkbar, config->toolbarBookmarks);
+}
+
+void on_action_toolbar_downloads_activate(GtkToggleAction* action, CBrowser* browser)
+{
+ /*config->toolbarDownloads = gtk_toggle_action_get_active(action);
+ sokoke_widget_set_visible(browser->downloadbar, config->toolbarDownloads);*/
+}
+
+void on_action_toolbar_status_activate(GtkToggleAction* action, CBrowser* browser)
+{
+ config->toolbarStatus = gtk_toggle_action_get_active(action);
+ sokoke_widget_set_visible(browser->statusbar, config->toolbarStatus);
+}
+
+void on_action_refresh_activate(GtkAction* action, CBrowser* browser)
+{
+ /*GdkModifierType state = (GdkModifierType)0;
+ gint x, y; gdk_window_get_pointer(NULL, &x, &y, &state);
+ gboolean fromCache = state & GDK_SHIFT_MASK;*/
+ webkit_web_view_reload(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)));
+}
+
+void on_action_refresh_stop_activate(GtkAction* action, CBrowser* browser)
+{
+ gchar* stockId; g_object_get(action, "stock-id", &stockId, NULL);
+ // Refresh or stop, depending on the stock id
+ if(!strcmp(stockId, GTK_STOCK_REFRESH))
+ {
+ /*GdkModifierType state = (GdkModifierType)0;
+ gint x, y; gdk_window_get_pointer(NULL, &x, &y, &state);
+ gboolean fromCache = state & GDK_SHIFT_MASK;*/
+ webkit_web_view_reload(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)));
+ }
+ else
+ webkit_web_view_stop_loading(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)));
+ g_free(stockId);
+}
+
+void on_action_stop_activate(GtkAction* action, CBrowser* browser)
+{
+ webkit_web_view_stop_loading(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)));
+}
+
+void on_action_zoom_in_activate(GtkAction* action, CBrowser* browser)
+{
+ /*GtkWidget* webView = get_nth_webView(-1, browser);
+ const gfloat zoom = webkit_web_view_get_text_multiplier(WEBKIT_WEB_VIEW(webView));
+ webkit_web_view_set_text_multiplier(WEBKIT_WEB_VIEW(webView), zoom + 0.1);*/
+}
+
+void on_action_zoom_out_activate(GtkAction* action, CBrowser* browser)
+{
+ /*GtkWidget* webView = get_nth_webView(-1, browser);
+ const gfloat zoom = webView_get_text_size(WEBKIT_WEB_VIEW(webView));
+ webkit_web_view_set_text_multiplier(WEBKIT_WEB_VIEW(webView), zoom - 0.1);*/
+}
+
+void on_action_zoom_normal_activate(GtkAction* action, CBrowser* browser)
+{
+ //webkit_web_view_set_text_multiplier(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)), 1);
+}
+
+void on_action_source_view_activate(GtkAction* action, CBrowser* browser)
+{
+ /*GtkWidget* webView = get_nth_webView(-1, browser);
+ gchar* source = webkit_web_view_copy_source(WEBKIT_WEB_VIEW(webView));
+ webkit_web_view_load_html_string(WEBKIT_WEB_VIEW(webView), source, "");
+ g_free(source);*/
+}
+
+void on_action_back_activate(GtkAction* action, CBrowser* browser)
+{
+ webkit_web_view_go_backward(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)));
+}
+
+void on_action_forward_activate(GtkAction* action, CBrowser* browser)
+{
+ webkit_web_view_go_forward(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)));
+}
+
+void on_action_home_activate(GtkAction* action, CBrowser* browser)
+{
+ webView_open(get_nth_webView(-1, browser), config->homepage);
+}
+
+void on_action_location_activate(GtkAction* action, CBrowser* browser)
+{
+ if(GTK_WIDGET_VISIBLE(browser->navibar))
+ gtk_widget_grab_focus(browser->location);
+ else
+ {
+ // TODO: We should offer all of the toolbar location's features here
+ GtkWidget* dialog;
+ dialog = gtk_dialog_new_with_buttons("Open location"
+ , GTK_WINDOW(browser->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR
+ , GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL
+ , GTK_STOCK_JUMP_TO, GTK_RESPONSE_ACCEPT
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog), GTK_STOCK_JUMP_TO);
+ gtk_container_set_border_width(GTK_CONTAINER(dialog), 5);
+ gtk_container_set_border_width(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), 5);
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ GtkWidget* label = gtk_label_new_with_mnemonic("_Location:");
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry), TRUE);
+ gtk_box_pack_start(GTK_BOX(hbox), entry, FALSE, FALSE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+ gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT);
+ if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
+ {
+ gtk_entry_set_text(GTK_ENTRY(browser->location)
+ , gtk_entry_get_text(GTK_ENTRY(entry)));
+ GdkEventKey event;
+ event.keyval = GDK_Return;
+ on_location_key_down(browser->location, &event, browser);
+ }
+ gtk_widget_destroy(dialog);
+ }
+}
+
+void on_action_webSearch_activate(GtkAction* action, CBrowser* browser)
+{
+ if(GTK_WIDGET_VISIBLE(browser->webSearch)
+ && GTK_WIDGET_VISIBLE(browser->navibar))
+ gtk_widget_grab_focus(browser->webSearch);
+ else
+ {
+ // TODO: We should offer all of the toolbar search's features here
+ GtkWidget* dialog = gtk_dialog_new_with_buttons("Web search"
+ , GTK_WINDOW(browser->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR
+ , GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL
+ , GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog), GTK_STOCK_FIND);
+ gtk_container_set_border_width(GTK_CONTAINER(dialog), 5);
+ gtk_container_set_border_width(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), 5);
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ GtkWidget* label = gtk_label_new_with_mnemonic("_Location:");
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry), TRUE);
+ gtk_box_pack_start(GTK_BOX(hbox), entry, FALSE, FALSE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+ gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT);
+ if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
+ {
+ gtk_entry_set_text(GTK_ENTRY(browser->webSearch)
+ , gtk_entry_get_text(GTK_ENTRY(entry)));
+ on_webSearch_activate(browser->webSearch, browser);
+ }
+ gtk_widget_destroy(dialog);
+ }
+}
+
+void on_menu_tabsClosed_activate(GtkWidget* widget, CBrowser* browser)
+{
+ GtkWidget* menu = gtk_menu_new();
+ guint n = xbel_folder_get_n_items(tabtrash);
+ GtkWidget* menuitem;
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ XbelItem* item = xbel_folder_get_nth_item(tabtrash, i);
+ const gchar* title = xbel_item_get_title(item);
+ const gchar* uri = xbel_bookmark_get_href(item);
+ menuitem = gtk_image_menu_item_new_with_label(title ? title : uri);
+ // FIXME: Get the real icon
+ GtkWidget* icon = gtk_image_new_from_stock(GTK_STOCK_FILE, GTK_ICON_SIZE_MENU);
+ gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem), icon);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ g_object_set_data(G_OBJECT(menuitem), "XbelItem", item);
+ g_signal_connect(menuitem, "activate", G_CALLBACK(on_menu_tabsClosed_item_activate), browser);
+ gtk_widget_show(menuitem);
+ }
+
+ menuitem = gtk_separator_menu_item_new();
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ gtk_widget_show(menuitem);
+ GtkAction* action = gtk_action_group_get_action(
+ browser->actiongroup, "TabsClosedClear");
+ menuitem = gtk_action_create_menu_item(action);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ gtk_widget_show(menuitem);
+ sokoke_widget_popup(widget, GTK_MENU(menu), NULL);
+}
+
+void on_menu_tabsClosed_item_activate(GtkWidget* menuitem, CBrowser* browser)
+{
+ // Create a new webView with an uri which has been closed before
+ XbelItem* item = g_object_get_data(G_OBJECT(menuitem), "XbelItem");
+ const gchar* uri = xbel_bookmark_get_href(item);
+ CBrowser* curBrowser = browser_new(browser);
+ webView_open(curBrowser->webView, uri);
+ xbel_folder_remove_item(tabtrash, item);
+ xbel_item_free(item);
+ update_browser_actions(curBrowser);
+}
+
+void on_action_tabsClosed_undo_activate(GtkAction* action, CBrowser* browser)
+{
+ // Open the most recent tabtrash item
+ XbelItem* item = xbel_folder_get_nth_item(tabtrash, 0);
+ const gchar* uri = xbel_bookmark_get_href(item);
+ CBrowser* curBrowser = browser_new(browser);
+ webView_open(curBrowser->webView, uri);
+ xbel_folder_remove_item(tabtrash, item);
+ xbel_item_free(item);
+ update_browser_actions(curBrowser);
+}
+
+void on_action_tabsClosed_clear_activate(GtkAction* action, CBrowser* browser)
+{
+ // Clear the closed tabs list
+ xbel_item_free(tabtrash);
+ tabtrash = xbel_folder_new();
+ update_browser_actions(browser);
+}
+
+void on_action_link_tab_new_activate(GtkAction* action, CBrowser* browser)
+{
+ CBrowser* curBrowser = browser_new(browser);
+ webView_open(curBrowser->webView, browser->elementUri);
+}
+
+void on_action_link_tab_current_activate(GtkAction* action, CBrowser* browser)
+{
+ webView_open(get_nth_webView(-1, browser), browser->elementUri);
+}
+
+void on_action_link_window_new_activate(GtkAction* action, CBrowser* browser)
+{
+ CBrowser* curBrowser = browser_new(NULL);
+ webView_open(curBrowser->webView, browser->elementUri);
+}
+
+void on_action_link_saveWith_activate(GtkAction* action, CBrowser* browser)
+{
+ spawn_protocol_command("download", browser->elementUri);
+}
+
+void on_action_link_copy_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkClipboard* clipboard = gtk_clipboard_get(GDK_SELECTION_CLIPBOARD);
+ gtk_clipboard_set_text(clipboard, browser->elementUri, -1);
+}
+
+static void browser_editBookmark_dialog_new(XbelItem* bookmark, CBrowser* browser)
+{
+ gboolean newBookmark = !bookmark;
+ GtkWidget* dialog = gtk_dialog_new_with_buttons(
+ newBookmark ? "New bookmark" : "Edit bookmark"
+ , GTK_WINDOW(browser->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR
+ , GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL
+ , newBookmark ? GTK_STOCK_ADD : GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog)
+ , newBookmark ? GTK_STOCK_ADD : GTK_STOCK_REMOVE);
+ gtk_container_set_border_width(GTK_CONTAINER(dialog), 5);
+ gtk_container_set_border_width(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), 5);
+ GtkSizeGroup* sizegroup = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
+
+ if(newBookmark)
+ bookmark = xbel_bookmark_new();
+
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ GtkWidget* label = gtk_label_new_with_mnemonic("_Title:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_title = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_title), TRUE);
+ if(!newBookmark)
+ gtk_entry_set_text(GTK_ENTRY(entry_title), xbel_item_get_title(bookmark));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_title, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ label = gtk_label_new_with_mnemonic("_Description:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_desc = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_desc), TRUE);
+ if(!newBookmark)
+ gtk_entry_set_text(GTK_ENTRY(entry_desc), xbel_item_get_desc(bookmark));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_desc, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ label = gtk_label_new_with_mnemonic("_Uri:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_uri = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_uri), TRUE);
+ if(!newBookmark)
+ gtk_entry_set_text(GTK_ENTRY(entry_uri), xbel_bookmark_get_href(bookmark));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_uri, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT);
+ if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
+ {
+ xbel_item_set_title(bookmark, gtk_entry_get_text(GTK_ENTRY(entry_title)));
+ xbel_item_set_desc(bookmark, gtk_entry_get_text(GTK_ENTRY(entry_desc)));
+ xbel_bookmark_set_href(bookmark, gtk_entry_get_text(GTK_ENTRY(entry_uri)));
+
+ // FIXME: We want to choose a folder
+ if(newBookmark)
+ xbel_folder_append_item(bookmarks, bookmark);
+ }
+ gtk_widget_destroy(dialog);
+}
+
+static void create_bookmark_menu(XbelItem*, GtkWidget*, CBrowser*);
+
+static void on_bookmark_menu_folder_activate(GtkWidget* menuitem, CBrowser* browser)
+{
+ GtkWidget* menu = gtk_menu_item_get_submenu(GTK_MENU_ITEM(menuitem));
+ gtk_container_foreach(GTK_CONTAINER(menu), (GtkCallback)gtk_widget_destroy, NULL);//...
+ XbelItem* folder = (XbelItem*)g_object_get_data(G_OBJECT(menuitem), "XbelItem");
+ create_bookmark_menu(folder, menu, browser);
+ // Remove all menuitems when the menu is hidden.
+ // FIXME: We really *want* the line below, but it won't work like that
+ //g_signal_connect_after(menu, "hide", G_CALLBACK(gtk_container_foreach), gtk_widget_destroy);
+ gtk_widget_show(menuitem);
+}
+
+static void on_bookmark_toolbar_folder_activate(GtkToolItem* toolitem, CBrowser* browser)
+{
+ GtkWidget* menu = gtk_menu_new();
+ XbelItem* folder = (XbelItem*)g_object_get_data(G_OBJECT(toolitem), "XbelItem");
+ create_bookmark_menu(folder, menu, browser);
+ // Remove all menuitems when the menu is hidden.
+ // FIXME: We really *should* run the line below, but it won't work like that
+ //g_signal_connect(menu, "hide", G_CALLBACK(gtk_container_foreach), gtk_widget_destroy);
+ sokoke_widget_popup(GTK_WIDGET(toolitem), GTK_MENU(menu), NULL);
+}
+
+void on_menu_bookmarks_item_activate(GtkWidget* widget, CBrowser* browser)
+{
+ XbelItem* item = (XbelItem*)g_object_get_data(G_OBJECT(widget), "XbelItem");
+ webView_open(get_nth_webView(-1, browser), xbel_bookmark_get_href(item));
+}
+
+static void create_bookmark_menu(XbelItem* folder, GtkWidget* menu, CBrowser* browser)
+{
+ guint n = xbel_folder_get_n_items(folder);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ XbelItem* item = xbel_folder_get_nth_item(folder, i);
+ const gchar* title = xbel_item_is_separator(item) ? "" : xbel_item_get_title(item);
+ //const gchar* desc = xbel_item_is_separator(item) ? "" : xbel_item_get_desc(item);
+ GtkWidget* menuitem = NULL;
+ switch(xbel_item_get_kind(item))
+ {
+ case XBEL_ITEM_FOLDER:
+ // FIXME: what about xbel_folder_is_folded?
+ menuitem = gtk_image_menu_item_new_with_label(title);
+ gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem)
+ , gtk_image_new_from_stock(GTK_STOCK_DIRECTORY, GTK_ICON_SIZE_MENU));
+ GtkWidget* _menu = gtk_menu_new();
+ gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), _menu);
+ g_signal_connect(menuitem, "activate"
+ , G_CALLBACK(on_bookmark_menu_folder_activate), browser);
+ g_object_set_data(G_OBJECT(menuitem), "XbelItem", item);
+ break;
+ case XBEL_ITEM_BOOKMARK:
+ menuitem = menu_item_new(title, STOCK_BOOKMARK
+ , G_CALLBACK(on_menu_bookmarks_item_activate), TRUE, browser);
+ g_object_set_data(G_OBJECT(menuitem), "XbelItem", item);
+ break;
+ case XBEL_ITEM_SEPARATOR:
+ menuitem = gtk_separator_menu_item_new();
+ break;
+ default:
+ g_warning("Unknown xbel item kind");
+ }
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ gtk_widget_show(menuitem);
+ }
+}
+
+void on_action_bookmark_new_activate(GtkAction* action, CBrowser* browser)
+{
+ browser_editBookmark_dialog_new(NULL, browser);
+}
+
+void on_action_manageSearchEngines_activate(GtkAction* action, CBrowser* browser)
+{
+ // Show the Manage search engines dialog. Create it if necessary.
+ static GtkWidget* dialog;
+ if(GTK_IS_DIALOG(dialog))
+ gtk_window_present(GTK_WINDOW(dialog));
+ else
+ {
+ dialog = webSearch_manageSearchEngines_dialog_new(browser);
+ gtk_widget_show(dialog);
+ }
+}
+
+void on_action_tab_previous_activate(GtkAction* action, CBrowser* browser)
+{
+ gint page = gtk_notebook_get_current_page(GTK_NOTEBOOK(browser->webViews));
+ gtk_notebook_set_current_page(GTK_NOTEBOOK(browser->webViews), page - 1);
+}
+
+void on_action_tab_next_activate(GtkAction* action, CBrowser* browser)
+{
+ // Advance one tab or jump to the first one if we are at the last one
+ gint page = gtk_notebook_get_current_page(GTK_NOTEBOOK(browser->webViews));
+ if(page == gtk_notebook_get_n_pages(GTK_NOTEBOOK(browser->webViews)) - 1)
+ page = -1;
+ gtk_notebook_set_current_page(GTK_NOTEBOOK(browser->webViews), page + 1);
+}
+
+void on_window_menu_item_activate(GtkImageMenuItem* widget, CBrowser* browser)
+{
+ gint page = get_webView_index(browser->webView, browser);
+ gtk_notebook_set_current_page(GTK_NOTEBOOK(browser->webViews), page);
+}
+
+void on_action_about_activate(GtkAction* action, CBrowser* browser)
+{
+ gtk_show_about_dialog(GTK_WINDOW(browser->window)
+ , "logo-icon-name", gtk_window_get_icon_name(GTK_WINDOW(browser->window))
+ , "name", PACKAGE_NAME
+ , "version", PACKAGE_VERSION
+ , "comments", "A lightweight web browser."
+ , "copyright", "Copyright © 2007 Christian Dywan"
+ , "website", "http://software.twotoasts.de"
+ , "authors", credits_authors
+ , "documenters", credits_documenters
+ , "artists", credits_artists
+ , "license", license
+ , "wrap-license", TRUE
+ //, "translator-credits", _("translator-credits")
+ , NULL);
+}
+
+gboolean on_location_key_down(GtkWidget* widget, GdkEventKey* event, CBrowser* browser)
+{
+ switch(event->keyval)
+ {
+ case GDK_Return:
+ {
+ const gchar* uri = gtk_entry_get_text(GTK_ENTRY(widget));
+ if(uri)
+ {
+ gchar* newUri = magic_uri(uri, TRUE);
+ // TODO: Use newUrl intermediately when completion is better
+ /* TODO Completion should be generated from history, that is
+ the uri as well as the title. */
+ entry_completion_append(GTK_ENTRY(widget), uri);
+ webView_open(get_nth_webView(-1, browser), newUri);
+ g_free(newUri);
+ }
+ return TRUE;
+ }
+ case GDK_Escape:
+ {
+ GtkWidget* webView = get_nth_webView(-1, browser);
+ WebKitWebFrame* frame = webkit_web_view_get_main_frame(WEBKIT_WEB_VIEW(webView));
+ const gchar* uri = webkit_web_frame_get_location(frame);
+ if(uri && *uri)
+ gtk_entry_set_text(GTK_ENTRY(widget), uri);
+ return TRUE;
+ }
+ }
+ return FALSE;
+}
+
+void on_location_changed(GtkWidget* widget, CBrowser* browser)
+{
+ // Preserve changes to the uri
+ /*const gchar* newUri = gtk_entry_get_text(GTK_ENTRY(widget));
+ xbel_bookmark_set_href(browser->sessionItem, newUri);*/
+ // FIXME: If we want this feature, this is the wrong approach
+}
+
+void on_action_panels_activate(GtkToggleAction* action, CBrowser* browser)
+{
+ config->panelShow = gtk_toggle_action_get_active(action);
+ sokoke_widget_set_visible(browser->panels, config->panelShow);
+}
+
+void on_action_panel_item_activate(GtkRadioAction* action
+ , GtkRadioAction* currentAction, CBrowser* browser)
+{
+ g_return_if_fail(GTK_IS_ACTION(action));
+ // TODO: Activating again should hide the contents; how?
+ //gint iValue; gint iCurrentValue;
+ //g_object_get(G_OBJECT(action), "value", &iValue, NULL);
+ //g_object_get(G_OBJECT(currentAction), "value", &iCurrentValue, NULL);
+ //GtkWidget* parent = gtk_widget_get_parent(browser->panels_notebook);
+ //sokoke_widget_set_visible(parent, iCurrentValue == iValue);
+ /*gtk_paned_set_position(GTK_PANED(gtk_widget_get_parent(browser->panels))
+ , iCurrentValue == iValue ? config->iPanelPos : 0);*/
+ config->panelActive = gtk_radio_action_get_current_value(action);
+ gint page = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(currentAction), "iPage"));
+ gtk_notebook_set_current_page(GTK_NOTEBOOK(browser->panels_notebook), page);
+ // This is a special case where activation was not user requested.
+ if(!GPOINTER_TO_INT(g_object_get_data(G_OBJECT(action), "once-silent")))
+ {
+ config->panelShow = TRUE;
+ gtk_widget_show(browser->panels);
+ }
+ else
+ g_object_set_data(G_OBJECT(action), "once-silent", NULL);
+}
+
+void on_action_openInPanel_activate(GtkAction* action, CBrowser* browser)
+{
+ GtkWidget* webView = get_nth_webView(-1, browser);
+ g_free(config->panelPageholder);
+ WebKitWebFrame* frame = webkit_web_view_get_main_frame(WEBKIT_WEB_VIEW(webView));
+ const gchar* uri = webkit_web_frame_get_location(frame);
+ config->panelPageholder = g_strdup(uri);
+ GtkAction* action_pageholder =
+ gtk_action_group_get_action(browser->actiongroup, "PanelPageholder");
+ gint value; g_object_get(G_OBJECT(action_pageholder), "value", &value, NULL);
+ sokoke_radio_action_set_current_value(GTK_RADIO_ACTION(action_pageholder), value);
+ gtk_widget_show(browser->panels);
+ webView_open(browser->panel_pageholder, config->panelPageholder);
+}
+
+
+static void on_panels_notify_position(GObject* object, GParamSpec* arg1
+ , CBrowser* browser)
+{
+ config->winPanelPos = gtk_paned_get_position(GTK_PANED(object));
+}
+
+void on_panels_button_close_clicked(GtkWidget* widget, CBrowser* browser)
+{
+ config->panelShow = FALSE;
+ gtk_widget_hide(browser->panels);
+}
+
+gboolean on_notebook_tab_mouse_up(GtkWidget* widget, GdkEventButton* event
+ , CBrowser* browser)
+{
+ if(event->button == 1 && event->type == GDK_2BUTTON_PRESS)
+ {
+ // Toggle the label visibility on double click
+ GtkWidget* child = gtk_bin_get_child(GTK_BIN(widget));
+ GList* children = gtk_container_get_children(GTK_CONTAINER(child));
+ child = (GtkWidget*)g_list_nth_data(children, 1);
+ gboolean visible = gtk_widget_get_child_visible(GTK_WIDGET(child));
+ gtk_widget_set_child_visible(GTK_WIDGET(child), !visible);
+ gint a, b; sokoke_widget_get_text_size(browser->webView_name, "M", &a, &b);
+ gtk_widget_set_size_request(child, !visible
+ ? a * config->tabSize : 0, !visible ? -1 : 0);
+ g_list_free(children);
+ return TRUE;
+ }
+ else if(event->button == 2)
+ {
+ // Close the webView on middle click
+ webView_close(browser->webView, browser);
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+gboolean on_notebook_tab_close_clicked(GtkWidget* widget, CBrowser* browser)
+{
+ webView_close(browser->webView, browser);
+ return TRUE;
+}
+
+void on_notebook_switch_page(GtkWidget* widget, GtkNotebookPage* page
+ , guint page_num, CBrowser* browser)
+{
+ GtkWidget* webView = get_nth_webView(page_num, browser);
+ browser = get_browser_from_webView(webView);
+ const gchar* uri = xbel_bookmark_get_href(browser->sessionItem);
+ gtk_entry_set_text(GTK_ENTRY(browser->location), uri);
+ const gchar* title = xbel_item_get_title(browser->sessionItem);
+ const gchar* effectiveTitle = title ? title : uri;
+ gchar* windowTitle = g_strconcat(effectiveTitle, " - ", PACKAGE_NAME, NULL);
+ gtk_window_set_title(GTK_WINDOW(browser->window), windowTitle);
+ g_free(windowTitle);
+ update_favicon(browser);
+ update_security(browser);
+ update_gui_state(browser);
+ update_statusbar_text(browser);
+ update_feeds(browser);
+ update_search_engines(browser);
+}
+
+void on_findbox_button_close_clicked(GtkWidget* widget, CBrowser* browser)
+{
+ gtk_widget_hide(browser->findbox);
+}
+
+static gboolean on_window_configure(GtkWidget* widget, GdkEventConfigure* event
+ , CBrowser* browser)
+{
+ gtk_window_get_position(GTK_WINDOW(browser->window)
+ , &config->winLeft, &config->winTop);
+ return FALSE;
+}
+
+static void on_window_size_allocate(GtkWidget* widget, GtkAllocation* allocation
+ , CBrowser* browser)
+{
+ config->winWidth = allocation->width;
+ config->winHeight = allocation->height;
+}
+
+gboolean on_window_destroy(GtkWidget* widget, GdkEvent* event, CBrowser* browser)
+{
+ gboolean proceed = TRUE;
+ // TODO: What if there are multiple windows?
+ // TODO: Smart dialog, Ã la 'Session?: Save, Discard, Cancel'
+ // TODO: Pref startup: session, ask, homepage, blank <-- ask
+ // TODO: Pref quit: session, ask, none <-- ask
+
+ if(0 /*g_list_length(browser_list) > 1*/)
+ {
+ GtkDialog* dialog;
+ dialog = GTK_DIALOG(gtk_message_dialog_new(GTK_WINDOW(browser->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_WARNING, GTK_BUTTONS_YES_NO
+ , "There is more than one tab open. Do you want to close anyway?"));
+ gtk_window_set_title(GTK_WINDOW(dialog), PACKAGE_NAME);
+ gtk_dialog_set_default_response(dialog, GTK_RESPONSE_YES);
+ proceed = gtk_dialog_run(dialog) == GTK_RESPONSE_YES;
+ gtk_widget_destroy(GTK_WIDGET(dialog));
+ }
+ return !proceed;
+}
+
+// -- Browser creation begins here
+
+CBrowser* browser_new(CBrowser* oldBrowser)
+{
+ CBrowser* browser = g_new0(CBrowser, 1);
+ browsers = g_list_prepend(browsers, browser);
+ browser->sessionItem = xbel_bookmark_new();
+ xbel_item_set_title(browser->sessionItem, "about:blank");
+ xbel_folder_append_item(session, browser->sessionItem);
+
+ GtkWidget* scrolled;
+
+ if(!oldBrowser)
+ {
+
+ GtkWidget* label; GtkWidget* hbox;
+
+ // Setup the window metrics
+ browser->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
+ GdkScreen* screen = gtk_window_get_screen(GTK_WINDOW(browser->window));
+ const gint defaultWidth = (gint)gdk_screen_get_width(screen) / 1.7;
+ const gint defaultHeight = (gint)gdk_screen_get_height(screen) / 1.7;
+ if(config->rememberWinMetrics)
+ {
+ if(!config->winWidth && !config->winHeight)
+ {
+ config->winWidth = defaultWidth;
+ config->winHeight = defaultWidth;
+ }
+ if(config->winLeft && config->winTop)
+ gtk_window_move(GTK_WINDOW(browser->window)
+ , config->winLeft, config->winTop);
+ gtk_window_set_default_size(GTK_WINDOW(browser->window)
+ , config->winWidth, config->winHeight);
+ }
+ else
+ gtk_window_set_default_size(GTK_WINDOW(browser->window)
+ , defaultWidth, defaultHeight);
+ g_signal_connect(browser->window, "configure-event"
+ , G_CALLBACK(on_window_configure), browser);
+ g_signal_connect(browser->window, "size-allocate"
+ , G_CALLBACK(on_window_size_allocate), browser);
+ // FIXME: Use custom program icon
+ gtk_window_set_icon_name(GTK_WINDOW(browser->window), "web-browser");
+ gtk_window_set_title(GTK_WINDOW(browser->window), g_get_application_name());
+ gtk_window_add_accel_group(GTK_WINDOW(browser->window), accel_group);
+ g_signal_connect(browser->window, "delete-event"
+ , G_CALLBACK(on_window_destroy), browser);
+ GtkWidget* vbox = gtk_vbox_new(FALSE, 0);
+ gtk_container_add(GTK_CONTAINER(browser->window), vbox);
+ gtk_widget_show(vbox);
+
+ // Let us see some ui manager magic
+ browser->actiongroup = gtk_action_group_new("Browser");
+ gtk_action_group_add_actions(browser->actiongroup, entries, entries_n, browser);
+ gtk_action_group_add_toggle_actions(browser->actiongroup
+ , toggle_entries, toggle_entries_n, browser);
+ gtk_action_group_add_radio_actions(browser->actiongroup
+ , refreshevery_entries, refreshevery_entries_n
+ , 300, NULL/*G_CALLBACK(activate_refreshevery_period_action)*/, browser);
+ gtk_action_group_add_radio_actions(browser->actiongroup
+ , panel_entries, panel_entries_n, -1
+ , G_CALLBACK(on_action_panel_item_activate), browser);
+ GtkUIManager* ui_manager = gtk_ui_manager_new();
+ gtk_ui_manager_insert_action_group(ui_manager, browser->actiongroup, 0);
+ gtk_window_add_accel_group(GTK_WINDOW(browser->window)
+ , gtk_ui_manager_get_accel_group(ui_manager));
+
+ GError* error = NULL;
+ if(!gtk_ui_manager_add_ui_from_string(ui_manager, ui_markup, -1, &error))
+ {
+ // TODO: Should this be a message dialog? When does this happen?
+ g_message("User interface couldn't be created: %s", error->message);
+ g_error_free(error);
+ }
+
+ GtkAction* action;
+ // Make all actions except toplevel menus which lack a callback insensitive
+ // This will vanish once all actions are implemented
+ guint i;
+ for(i = 0; i < entries_n; i++)
+ {
+ action = gtk_action_group_get_action(browser->actiongroup, entries[i].name);
+ gtk_action_set_sensitive(action, entries[i].callback || !entries[i].tooltip);
+ }
+ for(i = 0; i < toggle_entries_n; i++)
+ {
+ action = gtk_action_group_get_action(browser->actiongroup
+ , toggle_entries[i].name);
+ gtk_action_set_sensitive(action, toggle_entries[i].callback != NULL);
+ }
+ for(i = 0; i < refreshevery_entries_n; i++)
+ {
+ action = gtk_action_group_get_action(browser->actiongroup
+ , refreshevery_entries[i].name);
+ gtk_action_set_sensitive(action, FALSE);
+ }
+
+ //action_set_active("ToolbarDownloads", config->bToolbarDownloads, browser);
+
+ // Create the menubar
+ browser->menubar = gtk_ui_manager_get_widget(ui_manager, "/menubar");
+ GtkWidget* menuitem = gtk_menu_item_new();
+ gtk_widget_show(menuitem);
+ browser->throbber = gtk_image_new_from_stock(GTK_STOCK_EXECUTE, GTK_ICON_SIZE_MENU);
+ gtk_widget_show(browser->throbber);
+ gtk_container_add(GTK_CONTAINER(menuitem), browser->throbber);
+ gtk_widget_set_sensitive(menuitem, FALSE);
+ gtk_menu_item_set_right_justified(GTK_MENU_ITEM(menuitem), TRUE);
+ gtk_menu_shell_append(GTK_MENU_SHELL(browser->menubar), menuitem);
+ gtk_box_pack_start(GTK_BOX(vbox), browser->menubar, FALSE, FALSE, 0);
+ menuitem = gtk_ui_manager_get_widget(ui_manager, "/menubar/Go/TabsClosed");
+ g_signal_connect(menuitem, "activate"
+ , G_CALLBACK(on_menu_tabsClosed_activate), browser);
+ browser->menu_bookmarks = gtk_menu_item_get_submenu(
+ GTK_MENU_ITEM(gtk_ui_manager_get_widget(ui_manager, "/menubar/Bookmarks")));
+ menuitem = gtk_separator_menu_item_new();
+ gtk_widget_show(menuitem);
+ gtk_menu_shell_append(GTK_MENU_SHELL(browser->menu_bookmarks), menuitem);
+ browser->menu_window = gtk_menu_item_get_submenu(
+ GTK_MENU_ITEM(gtk_ui_manager_get_widget(ui_manager, "/menubar/Window")));
+ menuitem = gtk_separator_menu_item_new();
+ gtk_widget_show(menuitem);
+ gtk_menu_shell_append(GTK_MENU_SHELL(browser->menu_window), menuitem);
+ gtk_widget_show(browser->menubar);
+ action_set_sensitive("PrivateBrowsing", FALSE, browser); //...
+ action_set_sensitive("WorkOffline", FALSE, browser); //...
+ browser->popup_webView = gtk_ui_manager_get_widget(ui_manager, "/popup_webView");
+ g_object_ref(browser->popup_webView);
+ browser->popup_element = gtk_ui_manager_get_widget(ui_manager, "/popup_element");
+ g_object_ref(browser->popup_element);
+ browser->popup_editable = gtk_ui_manager_get_widget(ui_manager, "/popup_editable");
+ g_object_ref(browser->popup_editable);
+
+ // Create the navigation toolbar
+ browser->navibar = gtk_ui_manager_get_widget(ui_manager, "/toolbar_navigation");
+ gtk_toolbar_set_style(GTK_TOOLBAR(browser->navibar)
+ , config_to_toolbarstyle(config->toolbarStyle));
+ g_signal_connect(gtk_settings_get_default(), "notify::gtk-toolbar-style"
+ , G_CALLBACK(on_toolbar_navigation_notify_style), browser);
+ gtk_toolbar_set_icon_size(GTK_TOOLBAR(browser->navibar)
+ , config_to_toolbariconsize(config->toolbarSmall));
+ gtk_toolbar_set_show_arrow(GTK_TOOLBAR(browser->navibar), TRUE);
+ gtk_box_pack_start(GTK_BOX(vbox), browser->navibar, FALSE, FALSE, 0);
+ browser->newTab = gtk_ui_manager_get_widget(ui_manager, "/toolbar_navigation/TabNew");
+ action = gtk_action_group_get_action(browser->actiongroup, "Back");
+ g_object_set(action, "is-important", TRUE, NULL);
+
+ // Location entry
+ browser->location = sexy_icon_entry_new();
+ entry_setup_completion(GTK_ENTRY(browser->location));
+ sokoke_entry_set_can_undo(GTK_ENTRY(browser->location), TRUE);
+ browser->location_icon = gtk_image_new();
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->location)
+ , SEXY_ICON_ENTRY_PRIMARY, GTK_IMAGE(browser->location_icon));
+ sexy_icon_entry_add_clear_button(SEXY_ICON_ENTRY(browser->location));
+ g_signal_connect(browser->location, "key-press-event"
+ , G_CALLBACK(on_location_key_down), browser);
+ g_signal_connect(browser->location, "changed"
+ , G_CALLBACK(on_location_changed), browser);
+ GtkToolItem* toolitem = gtk_tool_item_new();
+ gtk_tool_item_set_expand(GTK_TOOL_ITEM(toolitem), TRUE);
+ gtk_container_add(GTK_CONTAINER(toolitem), browser->location);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->navibar), toolitem, -1);
+
+ // Search entry
+ browser->webSearch = sexy_icon_entry_new();
+ sexy_icon_entry_set_icon_highlight(SEXY_ICON_ENTRY(browser->webSearch)
+ , SEXY_ICON_ENTRY_PRIMARY, TRUE);
+ // TODO: Make this actively resizable or enlarge to fit contents?
+ // FIXME: The interface is somewhat awkward and ought to be rethought
+ // TODO: Display "show in context menu" search engines as "completion actions"
+ entry_setup_completion(GTK_ENTRY(browser->webSearch));
+ sokoke_entry_set_can_undo(GTK_ENTRY(browser->webSearch), TRUE);
+ update_searchEngine(config->searchEngine, browser);
+ g_signal_connect(browser->webSearch, "icon-released"
+ , G_CALLBACK(on_webSearch_icon_released), browser);
+ g_signal_connect(browser->webSearch, "key-press-event"
+ , G_CALLBACK(on_webSearch_key_down), browser);
+ g_signal_connect(browser->webSearch, "scroll-event"
+ , G_CALLBACK(on_webSearch_scroll), browser);
+ g_signal_connect(browser->webSearch, "activate"
+ , G_CALLBACK(on_webSearch_activate), browser);
+ toolitem = gtk_tool_item_new();
+ gtk_container_add(GTK_CONTAINER(toolitem), browser->webSearch);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->navibar), toolitem, -1);
+ action = gtk_action_group_get_action(browser->actiongroup, "TabsClosed");
+ browser->closedTabs = gtk_action_create_tool_item(action);
+ g_signal_connect(browser->closedTabs, "clicked"
+ , G_CALLBACK(on_menu_tabsClosed_activate), browser);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->navibar)
+ , GTK_TOOL_ITEM(browser->closedTabs), -1);
+ sokoke_container_show_children(GTK_CONTAINER(browser->navibar));
+ action_set_active("ToolbarNavigation", config->toolbarNavigation, browser);
+
+ // Bookmarkbar
+ browser->bookmarkbar = gtk_toolbar_new();
+ gtk_toolbar_set_icon_size(GTK_TOOLBAR(browser->bookmarkbar), GTK_ICON_SIZE_MENU);
+ gtk_toolbar_set_style(GTK_TOOLBAR(browser->bookmarkbar), GTK_TOOLBAR_BOTH_HORIZ);
+ create_bookmark_menu(bookmarks, browser->menu_bookmarks, browser);
+ for(i = 0; i < xbel_folder_get_n_items(bookmarks); i++)
+ {
+ XbelItem* item = xbel_folder_get_nth_item(bookmarks, i);
+ const gchar* title = xbel_item_is_separator(item)
+ ? "" : xbel_item_get_title(item);
+ const gchar* desc = xbel_item_is_separator(item)
+ ? "" : xbel_item_get_desc(item);
+ switch(xbel_item_get_kind(item))
+ {
+ case XBEL_ITEM_FOLDER:
+ toolitem = tool_button_new(title, GTK_STOCK_DIRECTORY, TRUE, TRUE
+ , G_CALLBACK(on_bookmark_toolbar_folder_activate), desc, browser);
+ g_object_set_data(G_OBJECT(toolitem), "XbelItem", item);
+ break;
+ case XBEL_ITEM_BOOKMARK:
+ toolitem = tool_button_new(title, STOCK_BOOKMARK, TRUE, TRUE
+ , G_CALLBACK(on_menu_bookmarks_item_activate), desc, browser);
+ g_object_set_data(G_OBJECT(toolitem), "XbelItem", item);
+ break;
+ case XBEL_ITEM_SEPARATOR:
+ toolitem = gtk_separator_tool_item_new();
+ break;
+ default:
+ g_warning("Unknown item kind");
+ }
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->bookmarkbar), toolitem, -1);
+ }
+ sokoke_container_show_children(GTK_CONTAINER(browser->bookmarkbar));
+ gtk_box_pack_start(GTK_BOX(vbox), browser->bookmarkbar, FALSE, FALSE, 0);
+ action_set_active("ToolbarBookmarks", config->toolbarBookmarks, browser);
+
+ // Superuser warning
+ if((hbox = sokoke_superuser_warning_new()))
+ gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
+
+ // Create the panels
+ GtkWidget* hpaned = gtk_hpaned_new();
+ gtk_paned_set_position(GTK_PANED(hpaned), config->winPanelPos);
+ g_signal_connect(hpaned, "notify::position"
+ , G_CALLBACK(on_panels_notify_position), browser);
+ gtk_box_pack_start(GTK_BOX(vbox), hpaned, TRUE, TRUE, 0);
+ gtk_widget_show(hpaned);
+
+ browser->panels = gtk_hbox_new(FALSE, 0);
+ gtk_paned_pack1(GTK_PANED(hpaned), browser->panels, FALSE, FALSE);
+ sokoke_widget_set_visible(browser->panels, config->panelShow);
+
+ // Create the panel toolbar
+ GtkWidget* panelbar = gtk_ui_manager_get_widget(ui_manager, "/toolbar_panels");
+ gtk_toolbar_set_style(GTK_TOOLBAR(panelbar), GTK_TOOLBAR_BOTH);
+ gtk_toolbar_set_icon_size(GTK_TOOLBAR(panelbar), GTK_ICON_SIZE_BUTTON);
+ gtk_toolbar_set_orientation(GTK_TOOLBAR(panelbar), GTK_ORIENTATION_VERTICAL);
+ gtk_box_pack_start(GTK_BOX(browser->panels), panelbar, FALSE, FALSE, 0);
+ action_set_active("Panels", config->panelShow, browser);
+ g_object_unref(ui_manager);
+
+ GtkWidget* cbox = gtk_vbox_new(FALSE, 0);
+ gtk_box_pack_start(GTK_BOX(browser->panels), cbox, TRUE, TRUE, 0);
+ gtk_widget_show(cbox);
+
+ // Panels titlebar
+ GtkWidget* labelbar = gtk_toolbar_new();
+ gtk_toolbar_set_icon_size(GTK_TOOLBAR(labelbar), GTK_ICON_SIZE_MENU);
+ gtk_toolbar_set_style(GTK_TOOLBAR(labelbar), GTK_TOOLBAR_ICONS);
+ toolitem = gtk_tool_item_new();
+ gtk_tool_item_set_expand(toolitem, TRUE);
+ label = gtk_label_new_with_mnemonic("_Panels");
+ gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
+ gtk_container_add(GTK_CONTAINER(toolitem), label);
+ gtk_container_set_border_width(GTK_CONTAINER(toolitem), 6);
+ gtk_toolbar_insert(GTK_TOOLBAR(labelbar), toolitem, -1);
+ // TODO: Does 'goto top' actually indicate 'detach'?
+ toolitem = tool_button_new(NULL, GTK_STOCK_GOTO_TOP, FALSE, TRUE
+ , NULL/*G_CALLBACK(on_panels_button_float_clicked)*/, "Detach panel", browser);
+ gtk_toolbar_insert(GTK_TOOLBAR(labelbar), toolitem, -1);
+ toolitem = tool_button_new(NULL, GTK_STOCK_CLOSE, FALSE, TRUE
+ , G_CALLBACK(on_panels_button_close_clicked), "Close panel", browser);
+ gtk_toolbar_insert(GTK_TOOLBAR(labelbar), toolitem, -1);
+ gtk_box_pack_start(GTK_BOX(cbox), labelbar, FALSE, FALSE, 0);
+ gtk_widget_show_all(labelbar);
+
+ // Notebook, containing all panels
+ browser->panels_notebook = gtk_notebook_new();
+ gtk_notebook_set_show_border(GTK_NOTEBOOK(browser->panels_notebook), FALSE);
+ gtk_notebook_set_show_tabs(GTK_NOTEBOOK(browser->panels_notebook), FALSE);
+ gint page;
+ // Dummy: This is the "fallback" panel for now
+ page = gtk_notebook_append_page(GTK_NOTEBOOK(browser->panels_notebook)
+ , gtk_label_new("empty"), NULL);
+ // Pageholder
+ browser->panel_pageholder = webView_new(&scrolled);
+ page = gtk_notebook_append_page(GTK_NOTEBOOK(browser->panels_notebook)
+ , scrolled, NULL);
+ //webView_load_from_uri(browser->panel_pageholder, config->panelPageholder);
+ action = gtk_action_group_get_action(browser->actiongroup, "PanelPageholder");
+ g_object_set_data(G_OBJECT(action), "iPage", GINT_TO_POINTER(page));
+ GtkWidget* frame = gtk_frame_new(NULL);
+ gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_IN);
+ gtk_container_add(GTK_CONTAINER(frame), browser->panels_notebook);
+ gtk_box_pack_start(GTK_BOX(cbox), frame, TRUE, TRUE, 0);
+ gtk_widget_show_all(gtk_widget_get_parent(browser->panels_notebook));
+ action = gtk_action_group_get_action(browser->actiongroup, "PanelDownloads");
+ g_object_set_data(G_OBJECT(action), "once-silent", GINT_TO_POINTER(1));
+ sokoke_radio_action_set_current_value(GTK_RADIO_ACTION(action), config->panelActive);
+ sokoke_widget_set_visible(browser->panels, config->panelShow);
+
+ // Notebook, containing all webViews
+ browser->webViews = gtk_notebook_new();
+ gtk_notebook_set_scrollable(GTK_NOTEBOOK(browser->webViews), TRUE);
+ #if GTK_CHECK_VERSION(2, 10, 0)
+ //gtk_notebook_set_group_id(GTK_NOTEBOOK(browser->webViews), 0);
+ #endif
+ gtk_paned_pack2(GTK_PANED(hpaned), browser->webViews, FALSE, FALSE);
+ gtk_widget_show(browser->webViews);
+
+ // Incremental findbar
+ browser->findbox = gtk_toolbar_new();
+ gtk_toolbar_set_icon_size(GTK_TOOLBAR(browser->findbox), GTK_ICON_SIZE_MENU);
+ gtk_toolbar_set_style(GTK_TOOLBAR(browser->findbox), GTK_TOOLBAR_BOTH_HORIZ);
+ toolitem = gtk_tool_item_new();
+ gtk_container_set_border_width(GTK_CONTAINER(toolitem), 6);
+ gtk_container_add(GTK_CONTAINER(toolitem)
+ , gtk_label_new_with_mnemonic("_Inline find:"));
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), toolitem, -1);
+ browser->findbox_text = sexy_icon_entry_new();
+ GtkWidget* icon = gtk_image_new_from_stock(GTK_STOCK_FIND, GTK_ICON_SIZE_MENU);
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->findbox_text)
+ , SEXY_ICON_ENTRY_PRIMARY, GTK_IMAGE(icon));
+ sexy_icon_entry_add_clear_button(SEXY_ICON_ENTRY(browser->findbox_text));
+ sokoke_entry_set_can_undo(GTK_ENTRY(browser->findbox_text), TRUE);
+ g_signal_connect(browser->findbox_text, "activate"
+ , G_CALLBACK(on_action_find_next_activate), browser);
+ toolitem = gtk_tool_item_new();
+ gtk_container_add(GTK_CONTAINER(toolitem), browser->findbox_text);
+ gtk_tool_item_set_expand(GTK_TOOL_ITEM(toolitem), TRUE);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), toolitem, -1);
+ toolitem = tool_button_new(NULL, GTK_STOCK_GO_BACK, TRUE, TRUE
+ , G_CALLBACK(on_action_find_previous_activate), NULL, browser);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), toolitem, -1);
+ toolitem = tool_button_new(NULL, GTK_STOCK_GO_FORWARD, TRUE, TRUE
+ , G_CALLBACK(on_action_find_next_activate), NULL, browser);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), toolitem, -1);
+ browser->findbox_case = gtk_toggle_tool_button_new_from_stock(GTK_STOCK_SPELL_CHECK);
+ gtk_tool_button_set_label(GTK_TOOL_BUTTON(browser->findbox_case), "Match Case");
+ gtk_tool_item_set_is_important(GTK_TOOL_ITEM(browser->findbox_case), TRUE);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), browser->findbox_case, -1);
+ browser->findbox_highlight = gtk_toggle_tool_button_new_from_stock(GTK_STOCK_SELECT_ALL);
+ g_signal_connect(browser->findbox_highlight, "toggled"
+ , G_CALLBACK(on_findbox_highlight_toggled), browser);
+ gtk_tool_button_set_label(GTK_TOOL_BUTTON(browser->findbox_highlight), "Highlight Matches");
+ gtk_tool_item_set_is_important(GTK_TOOL_ITEM(browser->findbox_highlight), TRUE);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), browser->findbox_highlight, -1);
+ toolitem = gtk_separator_tool_item_new();
+ gtk_separator_tool_item_set_draw(GTK_SEPARATOR_TOOL_ITEM(toolitem), FALSE);
+ gtk_tool_item_set_expand(GTK_TOOL_ITEM(toolitem), TRUE);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), toolitem, -1);
+ toolitem = tool_button_new(NULL, GTK_STOCK_CLOSE, FALSE, TRUE
+ , G_CALLBACK(on_findbox_button_close_clicked), "Close Findbar", browser);
+ gtk_toolbar_insert(GTK_TOOLBAR(browser->findbox), toolitem, -1);
+ sokoke_container_show_children(GTK_CONTAINER(browser->findbox));
+ gtk_box_pack_start(GTK_BOX(vbox), browser->findbox, FALSE, FALSE, 0);
+
+ // Statusbar
+ // TODO: fix children overlapping statusbar border
+ browser->statusbar = gtk_statusbar_new();
+ gtk_box_pack_start(GTK_BOX(vbox), browser->statusbar, FALSE, FALSE, 0);
+ browser->progress = gtk_progress_bar_new();
+ // Setting the progressbar's height to 1 makes it fit in the statusbar
+ gtk_widget_set_size_request(browser->progress, -1, 1);
+ gtk_box_pack_start(GTK_BOX(browser->statusbar), browser->progress
+ , FALSE, FALSE, 3);
+ browser->icon_security = gtk_image_new();
+ gtk_box_pack_start(GTK_BOX(browser->statusbar)
+ , browser->icon_security, FALSE, FALSE, 0);
+ gtk_widget_show(browser->icon_security);
+ browser->icon_newsfeed = gtk_image_new_from_icon_name(STOCK_NEWSFEED
+ , GTK_ICON_SIZE_MENU);
+ gtk_box_pack_start(GTK_BOX(browser->statusbar)
+ , browser->icon_newsfeed, FALSE, FALSE, 0);
+ action_set_active("ToolbarStatus", config->toolbarStatus, browser);
+
+ }
+ else
+ {
+
+ browser->window = oldBrowser->window;
+ browser->actiongroup = oldBrowser->actiongroup;
+ browser->menubar = oldBrowser->menubar;
+ browser->menu_bookmarks = oldBrowser->menu_bookmarks;
+ browser->menu_window = oldBrowser->menu_window;
+ browser->popup_webView = oldBrowser->popup_webView;
+ browser->popup_element = oldBrowser->popup_element;
+ browser->popup_editable = oldBrowser->popup_editable;
+ browser->throbber = oldBrowser->throbber;
+ browser->navibar = oldBrowser->navibar;
+ browser->newTab = oldBrowser->newTab;
+ browser->location_icon = oldBrowser->location_icon;
+ browser->location = oldBrowser->location;
+ browser->webSearch = oldBrowser->webSearch;
+ browser->closedTabs = oldBrowser->closedTabs;
+ browser->bookmarkbar = oldBrowser->bookmarkbar;
+ browser->panels = oldBrowser->panels;
+ browser->panels_notebook = oldBrowser->panels_notebook;
+ browser->panel_pageholder = oldBrowser->panel_pageholder;
+ browser->webViews = oldBrowser->webViews;
+ browser->findbox = oldBrowser->findbox;
+ browser->findbox_case = oldBrowser->findbox_case;
+ browser->findbox_highlight = oldBrowser->findbox_highlight;
+ browser->statusbar = oldBrowser->statusbar;
+ browser->progress = oldBrowser->progress;
+ browser->icon_security = oldBrowser->icon_security;
+ browser->icon_newsfeed = oldBrowser->icon_newsfeed;
+
+ }
+
+ // Define some default values
+ browser->hasMenubar = TRUE;
+ browser->hasToolbar = TRUE;
+ browser->hasLocation = TRUE;
+ browser->hasStatusbar = TRUE;
+ browser->elementUri = NULL;
+ browser->loadedPercent = -1; // initially "not loading"
+
+ // Add a window menu item
+ // TODO: Menu items should be ordered like the notebook tabs
+ // TODO: Watch tab reordering in >= gtk 2.10
+ browser->webView_menu = menu_item_new("about:blank", GTK_STOCK_FILE
+ , G_CALLBACK(on_window_menu_item_activate), TRUE, browser);
+ gtk_widget_show(browser->webView_menu);
+ gtk_menu_shell_append(GTK_MENU_SHELL(browser->menu_window), browser->webView_menu);
+
+ // Create a new tab label
+ GtkWidget* eventbox = gtk_event_box_new();
+ gtk_event_box_set_visible_window(GTK_EVENT_BOX(eventbox), FALSE);
+ g_signal_connect(eventbox, "button-release-event"
+ , G_CALLBACK(on_notebook_tab_mouse_up), browser);
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 1);
+ gtk_container_add(GTK_CONTAINER(eventbox), GTK_WIDGET(hbox));
+ browser->webView_icon = gtk_image_new_from_stock(GTK_STOCK_FILE, GTK_ICON_SIZE_MENU);
+ gtk_box_pack_start(GTK_BOX(hbox), browser->webView_icon, FALSE, FALSE, 0);
+ browser->webView_name = gtk_label_new(xbel_item_get_title(browser->sessionItem));
+ gtk_misc_set_alignment(GTK_MISC(browser->webView_name), 0.0, 0.5);
+ // TODO: make the tab initially look "unvisited" until it's focused
+ // TODO: tabs should shrink when there is not enough space
+ // TODO: gtk's tab scrolling is weird?
+ gint w, h;
+ sokoke_widget_get_text_size(browser->webView_name, "M", &w, &h);
+ gtk_widget_set_size_request(GTK_WIDGET(browser->webView_name)
+ , w * config->tabSize, -1);
+ gtk_label_set_ellipsize(GTK_LABEL(browser->webView_name), PANGO_ELLIPSIZE_END);
+ gtk_box_pack_start(GTK_BOX(hbox), browser->webView_name, FALSE, FALSE, 0);
+ browser->webView_close = gtk_button_new();
+ gtk_button_set_relief(GTK_BUTTON(browser->webView_close), GTK_RELIEF_NONE);
+ gtk_button_set_focus_on_click(GTK_BUTTON(browser->webView_close), FALSE);
+ GtkRcStyle* rcstyle = gtk_rc_style_new();
+ rcstyle->xthickness = rcstyle->ythickness = 0;
+ gtk_widget_modify_style(browser->webView_close, rcstyle);
+ GtkWidget* image = gtk_image_new_from_stock(GTK_STOCK_CLOSE, GTK_ICON_SIZE_MENU);
+ gtk_button_set_image(GTK_BUTTON(browser->webView_close), image);
+ gtk_box_pack_start(GTK_BOX(hbox), browser->webView_close, FALSE, FALSE, 0);
+ GtkSettings* gtksettings = gtk_settings_get_default();
+ gint height;
+ gtk_icon_size_lookup_for_settings(gtksettings, GTK_ICON_SIZE_BUTTON, 0, &height);
+ gtk_widget_set_size_request(browser->webView_close, -1, height);
+ gtk_widget_show_all(GTK_WIDGET(eventbox));
+ sokoke_widget_set_visible(browser->webView_close, config->tabClose);
+ g_signal_connect(browser->webView_close, "clicked"
+ , G_CALLBACK(on_notebook_tab_close_clicked), browser);
+
+ // Create a webView inside a scrolled window
+ browser->webView = webView_new(&scrolled);
+ gtk_widget_show(GTK_WIDGET(scrolled));
+ gtk_widget_show(GTK_WIDGET(browser->webView));
+ gint page = gtk_notebook_get_current_page(GTK_NOTEBOOK(browser->webViews));
+ page = gtk_notebook_insert_page(GTK_NOTEBOOK(browser->webViews)
+ , scrolled, GTK_WIDGET(eventbox), page + 1);
+ g_signal_connect_after(GTK_OBJECT(browser->webViews), "switch-page"
+ , G_CALLBACK(on_notebook_switch_page), browser);
+ #if GTK_CHECK_VERSION(2, 10, 0)
+ gtk_notebook_set_tab_reorderable(GTK_NOTEBOOK(browser->webViews), scrolled, TRUE);
+ gtk_notebook_set_tab_detachable(GTK_NOTEBOOK(browser->webViews), scrolled, TRUE);
+ #endif
+
+ // Connect signals
+ #define DOC_CONNECT(__sig, __func) g_signal_connect \
+ (G_OBJECT(browser->webView), __sig, G_CALLBACK(__func), browser);
+ DOC_CONNECT ("navigation-requested" , on_webView_navigation_requested)
+ DOC_CONNECT ("title-changed" , on_webView_title_changed)
+ DOC_CONNECT ("icon-loaded" , on_webView_icon_changed)
+ DOC_CONNECT ("load-started" , on_webView_load_started)
+ DOC_CONNECT ("load-progress-changed" , on_webView_load_changed)
+ DOC_CONNECT ("load-finished" , on_webView_load_finished)
+ DOC_CONNECT ("status-bar-text-changed" , on_webView_status_message)
+ DOC_CONNECT ("hovering-over-link" , on_webView_link_hover)
+ DOC_CONNECT ("console-message" , on_webView_console_message)
+
+ // For now we check for "plugins-enabled", in case this build has no properties
+ if(g_object_class_find_property(G_OBJECT_GET_CLASS(browser->webView), "plugins-enabled"))
+ g_object_set(G_OBJECT(browser->webView)
+ , "loads-images-automatically" , config->loadImagesAutomatically
+ , "shrinks-standalone-images-to-fit", config->shrinkImagesToFit
+ , "text-areas-are-resizable" , config->resizableTextAreas
+ , "java-script-enabled" , config->enableJavaScript
+ , "plugins-enabled" , config->enablePlugins
+ , NULL);
+
+ DOC_CONNECT ("button-release-event" , on_webView_button_release)
+ DOC_CONNECT ("popup-menu" , on_webView_popup);
+ DOC_CONNECT ("scroll-event" , on_webView_scroll);
+ DOC_CONNECT ("leave-notify-event" , on_webView_leave)
+ DOC_CONNECT ("destroy" , on_webView_destroy)
+ #undef DOC_CONNECT
+
+ // Eventually pack and display everything
+ sokoke_widget_set_visible(browser->navibar, config->toolbarNavigation);
+ sokoke_widget_set_visible(browser->newTab, config->toolbarNewTab);
+ sokoke_widget_set_visible(browser->webSearch, config->toolbarWebSearch);
+ sokoke_widget_set_visible(browser->closedTabs, config->toolbarClosedTabs);
+ sokoke_widget_set_visible(browser->bookmarkbar, config->toolbarBookmarks);
+ sokoke_widget_set_visible(browser->statusbar, config->toolbarStatus);
+ if(!config->openTabsInTheBackground)
+ gtk_notebook_set_current_page(GTK_NOTEBOOK(browser->webViews), page);
+
+ update_browser_actions(browser);
+ gtk_widget_show(browser->window);
+ gtk_widget_grab_focus(GTK_WIDGET(browser->location));
+
+ return browser;
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __BROWSER_H__
+#define __BROWSER_H__ 1
+
+#include "global.h"
+
+#include <gtk/gtk.h>
+
+// -- Types
+
+typedef struct _CBrowser
+{
+ // shared widgets
+ GtkWidget* window;
+ GtkActionGroup* actiongroup;
+ // menus
+ GtkWidget* menubar;
+ GtkWidget* menu_bookmarks;
+ GtkWidget* menu_window;
+ GtkWidget* popup_webView;
+ GtkWidget* popup_element;
+ GtkWidget* popup_editable;
+ GtkWidget* throbber;
+ // navibar
+ GtkWidget* navibar;
+ GtkWidget* newTab;
+ GtkWidget* location_icon;
+ GtkWidget* location;
+ GtkWidget* webSearch;
+ GtkWidget* closedTabs;
+ GtkWidget* bookmarkbar;
+ // panels
+ GtkWidget* panels;
+ GtkWidget* panels_notebook;
+ GtkWidget* panel_pageholder;
+ GtkWidget* webViews;
+ // findbox
+ GtkWidget* findbox;
+ GtkWidget* findbox_text;
+ GtkToolItem* findbox_case;
+ GtkToolItem* findbox_highlight;
+ GtkWidget* statusbar;
+ GtkWidget* progress;
+ GtkWidget* icon_security;
+ GtkWidget* icon_newsfeed;
+
+ // view specific widgets
+ GtkWidget* webView_menu;
+ GtkWidget* webView_icon;
+ GtkWidget* webView_name;
+ GtkWidget* webView_close;
+ GtkWidget* webView;
+
+ // view specific values
+ gboolean hasMenubar;
+ gboolean hasToolbar;
+ gboolean hasLocation;
+ gboolean hasStatusbar;
+ gchar* elementUri; // the element the mouse is hovering on
+ gint loadedPercent; // -1 means "not loading"
+ gint loadedBytes;
+ gint loadedBytesMax;
+ //UNDEFINED favicon;
+ guint security;
+ gchar* statusMessage; // message from a webView
+ XbelItem* sessionItem;
+} CBrowser;
+
+enum
+{
+ SEARCH_COL_ICON,
+ SEARCH_COL_TEXT,
+ SEARCH_COL_N
+};
+
+// -- Declarations
+
+void
+on_action_window_new_activate(GtkAction*, CBrowser*);
+
+void
+on_action_tab_new_activate(GtkAction*, CBrowser*);
+
+void
+on_action_open_activate(GtkAction*, CBrowser*);
+
+void
+on_action_tab_close_activate(GtkAction*, CBrowser*);
+
+void
+on_action_window_close_activate(GtkAction*, CBrowser*);
+
+void
+on_action_quit_activate(GtkAction*, CBrowser*);
+
+void
+on_action_edit_activate(GtkAction*, CBrowser*);
+
+void
+on_action_cut_activate(GtkAction*, CBrowser*);
+
+void
+on_action_copy_activate(GtkAction*, CBrowser*);
+
+void
+on_action_paste_activate(GtkAction*, CBrowser*);
+
+void
+on_action_delete_activate(GtkAction*, CBrowser*);
+
+void
+on_action_selectAll_activate(GtkAction*, CBrowser*);
+
+void
+on_action_find_activate(GtkAction*, CBrowser*);
+
+void
+on_action_find_next_activate(GtkAction*, CBrowser*);
+
+void
+on_action_find_previous_activate(GtkAction*, CBrowser*);
+
+void
+on_action_preferences_activate(GtkAction*, CBrowser*);
+
+void
+on_action_toolbar_navigation_activate(GtkToggleAction*, CBrowser*);
+
+void
+on_action_toolbar_bookmarks_activate(GtkToggleAction*, CBrowser*);
+
+void
+on_action_panels_activate(GtkToggleAction*, CBrowser*);
+
+void
+on_action_toolbar_status_activate(GtkToggleAction*, CBrowser*);
+
+void
+on_action_refresh_stop_activate(GtkAction*, CBrowser*);
+
+void
+on_action_zoom_in_activate(GtkAction*, CBrowser*);
+
+void
+on_action_zoom_out_activate(GtkAction*, CBrowser*);
+
+void
+on_action_zoom_normal_activate(GtkAction*, CBrowser*);
+
+void
+on_action_source_view_activate(GtkAction*, CBrowser*);
+
+void
+on_action_back_activate(GtkAction*, CBrowser*);
+
+void
+on_action_forward_activate(GtkAction*, CBrowser*);
+
+void
+on_action_home_activate(GtkAction*, CBrowser*);
+
+void
+on_action_location_activate(GtkAction*, CBrowser*);
+
+void
+on_action_webSearch_activate(GtkAction*, CBrowser*);
+
+void
+on_action_openInPanel_activate(GtkAction*, CBrowser*);
+
+void
+on_menu_tabsClosed_activate(GtkWidget*, CBrowser*);
+
+void
+on_menu_tabsClosed_item_activate(GtkWidget*, CBrowser*);
+
+void
+on_action_tabsClosed_clear_activate(GtkAction*, CBrowser*);
+
+void
+on_action_tabsClosed_undo_activate(GtkAction*, CBrowser*);
+
+void
+on_action_link_tab_new_activate(GtkAction*, CBrowser*);
+
+void
+on_action_link_tab_current_activate(GtkAction*, CBrowser*);
+
+void
+on_action_link_window_new_activate(GtkAction*, CBrowser*);
+
+void
+on_action_link_saveWith_activate(GtkAction*, CBrowser*);
+
+void
+on_action_link_copy_activate(GtkAction*, CBrowser*);
+
+void
+on_menu_bookmarks_item_activate(GtkWidget*, CBrowser*);
+
+void
+on_action_bookmark_new_activate(GtkAction*, CBrowser*);
+
+void
+on_action_manageSearchEngines_activate(GtkAction*, CBrowser*);
+
+void
+on_action_tab_previous_activate(GtkAction*, CBrowser*);
+
+void
+on_action_tab_next_activate(GtkAction*, CBrowser*);
+
+void
+on_action_about_activate(GtkAction*, CBrowser*);
+
+gboolean
+on_location_key_down(GtkWidget*, GdkEventKey*, CBrowser*);
+
+CBrowser*
+browser_new(CBrowser*);
+
+// -- Action definitions
+
+// TODO: Fill in a good description for each 'hm?'
+static const GtkActionEntry entries[] = {
+ { "File", NULL, "_File" },
+ { "WindowNew", STOCK_WINDOW_NEW
+ , NULL, "<Ctrl>n"
+ , "Open a new window", G_CALLBACK(on_action_window_new_activate) },
+ { "TabNew", STOCK_TAB_NEW
+ , NULL, "<Ctrl>t"
+ , "Open a new tab", G_CALLBACK(on_action_tab_new_activate) },
+ { "Open", GTK_STOCK_OPEN
+ , NULL, "<Ctrl>o"
+ , "Open a file", G_CALLBACK(on_action_open_activate) },
+ { "SaveAs", GTK_STOCK_SAVE_AS
+ , NULL, "<Ctrl>s"
+ , "Save to a file", NULL/*G_CALLBACK(on_action_saveas_activate)*/ },
+ { "TabClose", STOCK_TAB_CLOSE
+ , NULL, "<Ctrl>w"
+ , "Close the current tab", G_CALLBACK(on_action_tab_close_activate) },
+ { "WindowClose", STOCK_WINDOW_CLOSE
+ , NULL, "<Ctrl><Shift>w"
+ , "Close this window", G_CALLBACK(on_action_window_close_activate) },
+ { "PageSetup", GTK_STOCK_PROPERTIES
+ , "Pa_ge Setup", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_page_setup_activate)*/ },
+ { "PrintPreview", GTK_STOCK_PRINT_PREVIEW
+ , NULL, ""
+ , "hm?", NULL/*G_CALLBACK(on_action_print_preview_activate)*/ },
+ { "Print", GTK_STOCK_PRINT
+ , NULL, "<Ctrl>p"
+ , "hm?", NULL/*G_CALLBACK(on_action_print_activate)*/ },
+ { "Quit", GTK_STOCK_QUIT
+ , NULL, "<Ctrl>q"
+ , "Quit the application", G_CALLBACK(on_action_quit_activate) },
+
+ { "Edit", NULL, "_Edit", NULL, NULL, G_CALLBACK(on_action_edit_activate) },
+ { "Undo", GTK_STOCK_UNDO
+ , NULL, "<Ctrl>z"
+ , "Undo the last modification", NULL/*G_CALLBACK(on_action_undo_activate)*/ },
+ { "Redo", GTK_STOCK_REDO
+ , NULL, "<Ctrl><Shift>z"
+ , "Redo the last modification", NULL/*G_CALLBACK(on_action_redo_activate)*/ },
+ { "Cut", GTK_STOCK_CUT
+ , NULL, "<Ctrl>x"
+ , "Cut the selected text", G_CALLBACK(on_action_cut_activate) },
+ { "Copy", GTK_STOCK_COPY
+ , NULL, "<Ctrl>c"
+ , "Copy the selected text", G_CALLBACK(on_action_copy_activate) },
+ { "Copy_", GTK_STOCK_COPY
+ , NULL, "<Ctrl>c"
+ , "Copy the selected text", G_CALLBACK(on_action_copy_activate) },
+ { "Paste", GTK_STOCK_PASTE
+ , NULL, "<Ctrl>v"
+ , "Paste text from the clipboard", G_CALLBACK(on_action_paste_activate) },
+ { "Delete", GTK_STOCK_DELETE
+ , NULL, NULL
+ , "Delete the selected text", G_CALLBACK(on_action_delete_activate) },
+ { "SelectAll", GTK_STOCK_SELECT_ALL
+ , NULL, "<Ctrl>a"
+ , "Selected all text", G_CALLBACK(on_action_selectAll_activate) },
+ { "FormFill", STOCK_FORM_FILL
+ , NULL, ""
+ , "hm?", NULL/*G_CALLBACK(on_action_formfill_activate)*/ },
+ { "Find", GTK_STOCK_FIND
+ , NULL, "<Ctrl>f"
+ , "hm?", G_CALLBACK(on_action_find_activate) },
+ { "FindNext", GTK_STOCK_GO_FORWARD
+ , "Find _Next", "<Ctrl>g"
+ , "hm?", G_CALLBACK(on_action_find_next_activate) },
+ { "FindPrevious", GTK_STOCK_GO_BACK
+ , "Find _Previous", "<Ctrl><Shift>g"
+ , "hm?", G_CALLBACK(on_action_find_previous_activate) },
+ { "FindQuick", GTK_STOCK_FIND
+ , "_Quick Find", "period"
+ , "hm?", NULL/*G_CALLBACK(on_action_find_quick_activate)*/ },
+ { "ManageSearchEngines", GTK_STOCK_PROPERTIES
+ , "_Manage Search Engines", "<Ctrl><Alt>s"
+ , "hm?", G_CALLBACK(on_action_manageSearchEngines_activate) },
+ { "Preferences", GTK_STOCK_PREFERENCES
+ , NULL, "<Ctrl><Alt>p"
+ , "hm?", G_CALLBACK(on_action_preferences_activate) },
+
+ { "View", NULL, "_View" },
+ { "Toolbars", NULL, "_Toolbars" },
+ { "Refresh", GTK_STOCK_REFRESH
+ , NULL, "<Ctrl>r"
+ , "Refresh the current page", G_CALLBACK(on_action_refresh_stop_activate) },
+ // TODO: Is appointment-new a good choice?
+ // TODO: What if it isn't available?
+ { "RefreshEvery", "appointment-new"
+ , "Refresh _Every...", ""
+ , "Refresh the current page", G_CALLBACK(on_action_refresh_stop_activate) },
+ { "Stop", GTK_STOCK_STOP
+ , NULL, "Escape"
+ , "Stop loading of the current page", G_CALLBACK(on_action_refresh_stop_activate) },
+ { "RefreshStop", GTK_STOCK_REFRESH
+ , NULL, ""
+ , NULL, G_CALLBACK(on_action_refresh_stop_activate) },
+ { "ZoomIn", GTK_STOCK_ZOOM_IN
+ , NULL, "<Ctrl>plus"
+ , "hm?", G_CALLBACK(on_action_zoom_in_activate) },
+ { "ZoomOut", GTK_STOCK_ZOOM_OUT
+ , NULL, "<Ctrl>minus"
+ , "hm?", G_CALLBACK(on_action_zoom_out_activate) },
+ { "ZoomNormal", GTK_STOCK_ZOOM_100
+ , NULL, "<Ctrl>0"
+ , "hm?", G_CALLBACK(on_action_zoom_normal_activate) },
+ { "BackgroundImage", STOCK_IMAGE
+ , "_Background Image", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_background_image_activate)*/ },
+ { "SourceView", STOCK_SOURCE_VIEW
+ , NULL, ""
+ , "hm?", /*G_CALLBACK(on_action_source_view_activate)*/ },
+ { "SelectionSourceView", STOCK_SOURCE_VIEW
+ , "View Selection Source", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_selection_source_view_activate)*/ },
+ { "Properties", GTK_STOCK_PROPERTIES
+ , NULL, ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_activate)*/ },
+
+ { "Go", NULL, "_Go" },
+ { "Back", GTK_STOCK_GO_BACK
+ , NULL, "<Alt>Left"
+ , "hm?", G_CALLBACK(on_action_back_activate) },
+ { "Forward", GTK_STOCK_GO_FORWARD
+ , NULL, "<Alt>Right"
+ , "hm?", G_CALLBACK(on_action_forward_activate) },
+ { "Home", STOCK_HOMEPAGE
+ , NULL, "<Alt>Home"
+ , "hm?", G_CALLBACK(on_action_home_activate) },
+ { "Location", GTK_STOCK_JUMP_TO
+ , "Location...", "<Ctrl>l"
+ , "hm?", G_CALLBACK(on_action_location_activate) },
+ { "Websearch", GTK_STOCK_FIND
+ , "Websearch...", "<Ctrl><Shift>f"
+ , "hm?", G_CALLBACK(on_action_webSearch_activate) },
+ { "OpenInPageholder", GTK_STOCK_JUMP_TO
+ , "Open in Page_holder...", ""
+ , "hm?", G_CALLBACK(on_action_openInPanel_activate) },
+ { "TabsClosed", STOCK_USER_TRASH
+ , "Closed Tabs", ""
+ , "hm?", NULL },
+ { "TabsClosedClear", GTK_STOCK_CLEAR
+ , "Clear List of Closed Tabs", ""
+ , "hm?", G_CALLBACK(on_action_tabsClosed_clear_activate) },
+ { "UndoTabClose", GTK_STOCK_UNDELETE
+ , "Undo Close Tab", ""
+ , "hm?", G_CALLBACK(on_action_tabsClosed_undo_activate) },
+ { "LinkTabNew", STOCK_TAB_NEW
+ , "Open Link in New Tab", ""
+ , "hm?", G_CALLBACK(on_action_link_tab_new_activate) },
+ { "LinkTabCurrent", NULL
+ , "Open Link in Current Tab", ""
+ , "hm?", G_CALLBACK(on_action_link_tab_current_activate) },
+ { "LinkWindowNew", STOCK_WINDOW_NEW
+ , "Open Link in New Window", ""
+ , "hm?", G_CALLBACK(on_action_link_window_new_activate) },
+ { "LinkBookmarkNew", STOCK_BOOKMARK_NEW
+ , NULL, ""
+ , "Bookmark this link", NULL/*G_CALLBACK(on_action_link_bookmark_activate)*/ },
+ { "LinkSaveAs", GTK_STOCK_SAVE
+ , "Save Destination as...", ""
+ , "Save destination to a file", NULL/*G_CALLBACK(on_action_link_saveas_activate)*/ },
+ { "LinkSaveWith", STOCK_DOWNLOADS
+ , "Download Destination", ""
+ , "Save destination with the chosen download manager", G_CALLBACK(on_action_link_saveWith_activate) },
+ { "LinkCopy", GTK_STOCK_COPY
+ , "Copy Link Address", ""
+ , "Copy the link address to the clipboard", G_CALLBACK(on_action_link_copy_activate) },
+ { "SelectionLinksNewTabs", NULL
+ , "Open Selected Links in Tabs", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "SelectionTextTabNew", STOCK_TAB_NEW
+ , "Open <Selection> in New Tab", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "SelectionTextTabCurrent", NULL
+ , "Open <Selection> in Current Tab", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "SelectionTextWindowNew", STOCK_WINDOW_NEW
+ , "Open <Selection> in New Qindow", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "SelectionSearch", GTK_STOCK_FIND
+ , "Search for <Selection>", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "SelectionSearchWith", GTK_STOCK_FIND
+ , "Search for <Selection> with...", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "ImageViewTabNew", STOCK_TAB_NEW
+ , "View Image in New Tab", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "ImageViewTabCurrent", NULL
+ , "View image in current tab", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "ImageSaveAs", GTK_STOCK_SAVE
+ , "Save Image as...", ""
+ , "Save image to a file", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "ImageSaveWith", STOCK_DOWNLOADS
+ , "Download Image", ""
+ , "Save image with the chosen download manager", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+ { "ImageCopy", GTK_STOCK_COPY
+ , "Copy Image Address", ""
+ , "Copy the image address to the clipboard", NULL/*G_CALLBACK(on_action_properties_selection_activate)*/ },
+
+ { "Bookmarks", NULL, "_Bookmarks" },
+ { "BookmarkNew", STOCK_BOOKMARK_NEW
+ , NULL, "<Ctrl>d"
+ , "hm?", NULL/*G_CALLBACK(on_action_bookmark_new_activate)*/ },
+ { "BookmarksManage", STOCK_BOOKMARKS
+ , "_Manage Bookmarks", "<Ctrl>b"
+ , "hm?", NULL/*G_CALLBACK(on_action_bookmarks_manage_activate)*/ },
+
+ { "Tools", NULL, "_Tools" },
+
+ { "Window", NULL, "_Window" },
+ { "SessionLoad", GTK_STOCK_REVERT_TO_SAVED
+ , "_Load Session", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_session_load_activate)*/ },
+ { "SessionSave", GTK_STOCK_SAVE_AS
+ , "_Save Session", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_session_save_activate)*/ },
+ { "TabPrevious", GTK_STOCK_GO_BACK
+ , "_Previous Tab", "<Ctrl>Page_Up"
+ , "hm?", G_CALLBACK(on_action_tab_previous_activate) },
+ { "TabNext", GTK_STOCK_GO_FORWARD
+ , "_Next Tab", "<Ctrl>Page_Down"
+ , "hm?", G_CALLBACK(on_action_tab_next_activate) },
+ { "TabOverview", NULL
+ , "Tab _Overview", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_tab_overview_activate)*/ },
+
+ { "Help", NULL, "_Help" },
+ { "HelpContents", GTK_STOCK_HELP
+ , "_Contents", "F1"
+ , "hm?", NULL/*G_CALLBACK(on_action_help_contents_activate)*/ },
+ { "About", GTK_STOCK_ABOUT
+ , NULL, ""
+ , "hm?", G_CALLBACK(on_action_about_activate) },
+ };
+ static const guint entries_n = G_N_ELEMENTS(entries);
+
+static const GtkToggleActionEntry toggle_entries[] = {
+ { "PrivateBrowsing", NULL
+ , "P_rivate Browsing", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_private_browsing_activate)*/
+ , FALSE },
+ { "WorkOffline", GTK_STOCK_DISCONNECT
+ , "_Work Offline", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_work_offline_activate)*/
+ , FALSE },
+
+ { "ToolbarNavigation", NULL
+ , "_Navigationbar", ""
+ , "hm?", G_CALLBACK(on_action_toolbar_navigation_activate)
+ , FALSE },
+ { "Panels", NULL
+ , "_Panels", "F9"
+ , "hm?", G_CALLBACK(on_action_panels_activate)
+ , FALSE },
+ { "ToolbarBookmarks", NULL
+ , "_Bookmarkbar", ""
+ , "hm?", G_CALLBACK(on_action_toolbar_bookmarks_activate)
+ , FALSE },
+ { "ToolbarDownloads", NULL
+ , "_Downloadbar", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_toolbar_downloads_activate)*/
+ , FALSE },
+ { "ToolbarStatus", NULL
+ , "_Statusbar", ""
+ , "hm?", G_CALLBACK(on_action_toolbar_status_activate)
+ , FALSE },
+ { "RefreshEveryEnable", NULL
+ , "_Enabled", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_reloadevery_enable_activate)*/
+ , FALSE },
+ { "ReloadEveryActive", NULL
+ , "_Active", ""
+ , "hm?", NULL/*G_CALLBACK(on_action_reloadevery_active_activate)*/
+ , FALSE },
+ };
+ static const guint toggle_entries_n = G_N_ELEMENTS(toggle_entries);
+
+static const GtkRadioActionEntry refreshevery_entries[] = {
+ { "RefreshEvery30", NULL
+ , "30 seconds", ""
+ , "Refresh Every _30 Seconds", 30 },
+ { "RefreshEvery60", NULL
+ , "60 seconds", ""
+ , "Refresh Every _60 Seconds", 60 },
+ { "RefreshEvery300", NULL
+ , "5 minutes", ""
+ , "Refresh Every _5 Minutes", 300 },
+ { "RefreshEvery900", NULL
+ , "15 minutes", ""
+ , "Refresh Every _15 Minutes", 900 },
+ { "RefreshEvery1800", NULL
+ , "30 minutes", ""
+ , "Refresh Every 3_0 Minutes", 1800 },
+ { "RefreshEveryCustom", NULL
+ , "Custom...", ""
+ , "Refresh by a _Custom Period", 0 },
+ };
+ static const guint refreshevery_entries_n = G_N_ELEMENTS(refreshevery_entries);
+
+static const GtkRadioActionEntry panel_entries[] = {
+ { "PanelDownloads", STOCK_DOWNLOADS
+ , NULL, ""
+ , "hm?", 0 },
+ { "PanelBookmarks", STOCK_BOOKMARKS
+ , "_Bookmarks", ""
+ , "hm?", 1 },
+ { "PanelConsole", STOCK_CONSOLE
+ , NULL, ""
+ , "hm?", 2 },
+ { "PanelExtensions", STOCK_EXTENSIONS
+ , NULL, ""
+ , "hm?", 3 },
+ { "PanelHistory", STOCK_HISTORY
+ , "_History", ""
+ , "hm?", 4 },
+ // TODO: We want a better icon here, but which one?
+ { "PanelTabs", STOCK_TAB_NEW
+ , "_Tabs", ""
+ , "hm?", 5 },
+ // TODO: We probably want another icon here
+ { "PanelPageholder", GTK_STOCK_CONVERT
+ , "_Pageholder", ""
+ , "hm?", 6 },
+ };
+ static const guint panel_entries_n = G_N_ELEMENTS(panel_entries);
+
+#endif /* !__BROWSER_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "conf.h"
+
+#include "sokoke.h"
+
+#include <glib.h>
+#include <stdio.h>
+#include <string.h>
+
+CConfig* config_new(void)
+{
+ return g_new0(CConfig, 1);
+}
+
+void config_free(CConfig* config)
+{
+ g_free(config->homepage);
+ g_free(config->locationSearch);
+ g_free(config->panelPageholder);
+ g_datalist_clear(&config->protocols_commands);
+ g_ptr_array_free(config->protocols_names, TRUE);
+ g_free(config);
+}
+
+gboolean config_from_file(CConfig* config, const gchar* filename, GError** error)
+{
+ GKeyFile* keyFile = g_key_file_new();
+ g_key_file_load_from_file(keyFile, filename, G_KEY_FILE_KEEP_COMMENTS, error);
+ /*g_key_file_load_from_data_dirs(keyFile, sFilename, NULL
+ , G_KEY_FILE_KEEP_COMMENTS, error);*/
+ #define GET_INT(var, key, default) \
+ var = sokoke_key_file_get_integer_default( \
+ keyFile, "browser", key, default, NULL)
+ #define GET_STR(var, key, default) \
+ var = sokoke_key_file_get_string_default( \
+ keyFile, "browser", key, default, NULL)
+ GET_INT(config->startup, "Startup", CONFIG_STARTUP_HOMEPAGE);
+ GET_STR(config->homepage, "Homepage", "http://www.google.com");
+ GET_STR(config->locationSearch, "LocationSearch", "http://www.google.com/search?q=%s");
+ GET_INT(config->toolbarNavigation, "ToolbarNavigation", TRUE);
+ GET_INT(config->toolbarBookmarks, "ToolbarBookmarks", FALSE);
+ GET_INT(config->toolbarStatus, "ToolbarStatus", TRUE);
+ //GET_INT(config->toolbarDownloads, "ToolbarDownloads", FALSE);
+ GET_INT(config->toolbarStyle, "ToolbarStyle", CONFIG_TOOLBAR_DEFAULT);
+ GET_INT(config->toolbarSmall, "ToolbarSmall", FALSE);
+ GET_INT(config->toolbarWebSearch, "ToolbarWebSearch", TRUE);
+ GET_INT(config->toolbarNewTab, "ToolbarNewTab", TRUE);
+ GET_INT(config->toolbarClosedTabs, "ToolbarClosedTabs", TRUE);
+ GET_INT(config->panelShow, "PanelShow", FALSE);
+ GET_INT(config->panelActive, "PanelActive", 0);
+ GET_STR(config->panelPageholder, "PanelPageholder", "http://www.google.com");
+ GET_INT(config->tabSize, "TabSize", 10);
+ GET_INT(config->tabClose, "TabClose", TRUE);
+ GET_INT(config->newPages, "NewPages", CONFIG_NEWPAGES_TAB_NEW);
+ GET_INT(config->openTabsInTheBackground, "OpenTabsInTheBackground", FALSE);
+ GET_INT(config->openPopupsInTabs, "OpenPopupsInTabs", FALSE);
+ #undef GET_INT
+ #undef GET_STR
+
+ #define GET_INT(var, key, default) \
+ var = sokoke_key_file_get_integer_default( \
+ keyFile, "content", key, default, NULL)
+ #define GET_STR(var, key, default) \
+ var = sokoke_key_file_get_string_default( \
+ keyFile, "content", key, default, NULL)
+ GET_INT(config->loadImagesAutomatically, "LoadImagesAutomatically", TRUE);
+ GET_INT(config->shrinkImagesToFit, "ShrinkImagesTooFit", TRUE);
+ GET_INT(config->resizableTextAreas, "ResizableTextAreas", FALSE);
+ GET_INT(config->enableJavaScript, "EnableJavaScript", TRUE);
+ GET_INT(config->enablePlugins, "EnablePlugins", TRUE);
+ #undef GET_INT
+ #undef GET_STR
+
+ #define GET_INT(var, key, default) \
+ var = sokoke_key_file_get_integer_default( \
+ keyFile, "session", key, default, NULL)
+ #define GET_STR(var, key, default) \
+ var = sokoke_key_file_get_string_default( \
+ keyFile, "session", key, default, NULL)
+ GET_INT(config->rememberWinMetrics , "RememberWinMetrics", TRUE);
+ GET_INT(config->winLeft , "WinLeft", 0);
+ GET_INT(config->winTop , "WinTop", 0);
+ GET_INT(config->winWidth , "WinWidth", 0);
+ GET_INT(config->winHeight, "WinHeight", 0);
+ GET_INT(config->winPanelPos, "WinPanelPos", 0);
+ GET_INT(config->searchEngine, "SearchEngine", 0);
+ #undef GET_INT
+ #undef GET_STR
+
+ config->protocols_names = g_ptr_array_new();
+ g_datalist_init(&config->protocols_commands);
+ gchar** protocols;
+ if((protocols = g_key_file_get_keys(keyFile, "protocols", NULL, NULL)))
+ {
+ guint i;
+ for(i = 0; protocols[i] != NULL; i++)
+ {
+ gchar* sCommand = g_key_file_get_string(keyFile, "protocols"
+ , protocols[i], NULL);
+ g_ptr_array_add(config->protocols_names, (gpointer)protocols[i]);
+ g_datalist_set_data_full(&config->protocols_commands
+ , protocols[i], sCommand, g_free);
+ }
+ g_free(protocols);
+ }
+
+ g_key_file_free(keyFile);
+ return !(error && *error);
+}
+
+gboolean config_to_file(CConfig* config, const gchar* filename, GError** error)
+{
+ GKeyFile* keyFile = g_key_file_new();
+
+ g_key_file_set_integer(keyFile, "browser", "Startup", config->startup);
+ g_key_file_set_string (keyFile, "browser", "Homepage", config->homepage);
+ g_key_file_set_string (keyFile, "browser", "LocationSearch", config->locationSearch);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarNavigation", config->toolbarNavigation);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarBookmarks", config->toolbarBookmarks);
+ //g_key_file_set_integer(keyFile, "browser", "ToolbarDownloads", config->toolbarDownloads);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarStatus", config->toolbarStatus);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarStyle", config->toolbarStyle);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarSmall", config->toolbarSmall);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarWebSearch", config->toolbarWebSearch);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarNewTab", config->toolbarNewTab);
+ g_key_file_set_integer(keyFile, "browser", "ToolbarClosedTabs", config->toolbarClosedTabs);
+ g_key_file_set_integer(keyFile, "browser", "PanelShow", config->panelShow);
+ g_key_file_set_integer(keyFile, "browser", "PanelActive", config->panelActive);
+ g_key_file_set_string (keyFile, "browser", "PanelPageholder", config->panelPageholder);
+ g_key_file_set_integer(keyFile, "browser", "TabSize", config->tabSize);
+ g_key_file_set_integer(keyFile, "browser", "TabClose", config->tabClose);
+ g_key_file_set_integer(keyFile, "browser", "NewPages", config->newPages);
+ g_key_file_set_integer(keyFile, "browser", "OpenTabsInTheBackground", config->openTabsInTheBackground);
+ g_key_file_set_integer(keyFile, "browser", "OpenPopupsInTabs", config->openPopupsInTabs);
+
+ g_key_file_set_integer(keyFile, "content", "LoadImagesAutomatically", config->loadImagesAutomatically);
+ g_key_file_set_integer(keyFile, "content", "ShrinkImagesToFit", config->shrinkImagesToFit);
+ g_key_file_set_integer(keyFile, "content", "ResizableTextAreas", config->resizableTextAreas);
+ g_key_file_set_integer(keyFile, "content", "EnableJavaScript", config->enableJavaScript);
+ g_key_file_set_integer(keyFile, "content", "EnablePlugins", config->enablePlugins);
+
+ g_key_file_set_integer(keyFile, "session", "RememberWinMetrics", config->rememberWinMetrics);
+ g_key_file_set_integer(keyFile, "session", "WinLeft", config->winLeft);
+ g_key_file_set_integer(keyFile, "session", "WinTop", config->winTop);
+ g_key_file_set_integer(keyFile, "session", "WinWidth", config->winWidth);
+ g_key_file_set_integer(keyFile, "session", "WinHeight", config->winHeight);
+ g_key_file_set_integer(keyFile, "session", "WinPanelPos", config->winPanelPos);
+ g_key_file_set_integer(keyFile, "session", "SearchEngine", config->searchEngine);
+
+ guint i;
+ for(i = 0; i < config->protocols_names->len; i++)
+ {
+ gchar* protocol = (gchar*)g_ptr_array_index(config->protocols_names, i);
+ gchar* command = g_datalist_get_data(&config->protocols_commands, protocol);
+ g_key_file_set_string(keyFile, "protocols", protocol, command);
+ g_free(protocol);
+ }
+
+ gboolean saved = sokoke_key_file_save_to_file(keyFile, filename, error);
+ g_key_file_free(keyFile);
+
+ return saved;
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __CONF_H__
+#define __CONF_H__ 1
+
+#include <glib.h>
+
+typedef struct _CConfig
+{
+ guint startup;
+ gchar* homepage;
+ gchar* locationSearch;
+ gboolean toolbarNavigation;
+ gboolean toolbarBookmarks;
+ //gboolean toolbarDownloads;
+ gboolean toolbarStatus;
+ guint toolbarStyle;
+ gboolean toolbarSmall;
+ gboolean panelShow;
+ guint panelActive;
+ gchar* panelPageholder;
+ // TODO: What about this? Support it or fix tab shrinking and drop it?
+ guint tabSize; // tab size in charcters
+ gboolean tabClose;
+ gboolean toolbarWebSearch;
+ gboolean toolbarNewTab;
+ gboolean toolbarClosedTabs;
+ guint newPages; // where to open new pages
+ gboolean openTabsInTheBackground;
+ gboolean openPopupsInTabs;
+
+ gboolean loadImagesAutomatically;
+ gboolean shrinkImagesToFit;
+ gboolean resizableTextAreas;
+ gboolean enableJavaScript;
+ gboolean enablePlugins;
+
+ gboolean rememberWinMetrics; // Restore last state upon startup?
+ gint winLeft;
+ gint winTop;
+ gint winWidth;
+ gint winHeight;
+ guint winPanelPos;
+ guint searchEngine; // last selected search engine
+
+ GPtrArray* protocols_names;
+ GData* protocols_commands;
+} CConfig;
+
+enum
+{
+ CONFIG_STARTUP_BLANK,
+ CONFIG_STARTUP_HOMEPAGE,
+ CONFIG_STARTUP_SESSION
+};
+
+enum
+{
+ CONFIG_TOOLBAR_DEFAULT,
+ CONFIG_TOOLBAR_ICONS,
+ CONFIG_TOOLBAR_TEXT,
+ CONFIG_TOOLBAR_BOTH,
+ CONFIG_TOOLBAR_BOTH_HORIZ
+};
+
+enum
+{
+ CONFIG_NEWPAGES_TAB_NEW,
+ CONFIG_NEWPAGES_WINDOW_NEW,
+ CONFIG_NEWPAGES_TAB_CURRENT
+};
+
+CConfig*
+config_new(void);
+
+void
+config_free(CConfig*);
+
+gboolean
+config_from_file(CConfig*, const gchar*, GError**);
+
+gboolean
+config_to_file(CConfig*, const gchar*, GError**);
+
+#endif /* !__CONF_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __DEBUG_H__
+#define __DEBUG_H__ 1
+
+#include "config.h"
+
+#include <glib/gprintf.h>
+
+#if SOKOKE_DEBUG > 1
+ #define UNIMPLEMENTED g_print(" * Unimplemented: %s\n", G_STRFUNC);
+#else
+ #define UNIMPLEMENTED ;
+#endif
+
+#endif /* !__DEBUG_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __GLOBAL_H__
+#define __GLOBAL_H__ 1
+
+#include "conf.h"
+#include "xbel.h"
+
+#include <gtk/gtk.h>
+
+// -- globals
+
+CConfig* config;
+GList* searchEngines; // Items of type 'SearchEngine'
+GList* browsers; // Items of type 'CBrowser'
+GtkAccelGroup* accel_group;
+XbelItem* bookmarks;
+XbelItem* session;
+XbelItem* tabtrash;
+
+// Custom stock items
+
+// We should distribute these
+// Names should match with epiphany and/ or xdg spec
+/* NOTE: Those uncommented were replaced with remotely related icons
+ in order to reduce the amount of warnings :D */
+
+#define STOCK_BOOKMARK GTK_STOCK_FILE // "stock_bookmark" // "bookmark-web"
+#define STOCK_BOOKMARKS "bookmark-view"
+#define STOCK_DOWNLOADS "package" // "download"
+#define STOCK_CONSOLE "terminal" // "console" // MISSING
+#define STOCK_EXTENSIONS "extension" // MISSING
+#define STOCK_FORM_FILL "insert-text" // "form-fill" // MISSING
+#define STOCK_HISTORY "document-open-recent"
+#define STOCK_HISTORY_ "history-view"
+#define STOCK_LOCATION "location-entry"
+#define STOCK_NEWSFEED "gtk-index" // "newsfeed" // MISSING
+#define STOCK_PLUGINS "plugin" // MISSING
+#define STOCK_POPUPS_BLOCKED "popup-hidden"
+#define STOCK_SOURCE_VIEW "stock_view-html-source" // MISSING
+#define STOCK_TAB_CLOSE "tab-close" // MISSING
+#define STOCK_WINDOW_CLOSE "window-close" // MISSING
+
+// We can safely use standard icons
+// Assuming that we have reliable fallback icons
+
+#define STOCK_BOOKMARK_NEW "bookmark-new"
+#define STOCK_BOOKMARK_NEW_ "stock_add-bookmark"
+#define STOCK_HOMEPAGE GTK_STOCK_HOME
+#define STOCK_IMAGE "image-x-generic"
+#define STOCK_IMAGE_ "gnome-mime-image"
+#define STOCK_LOCK_OPEN "stock_lock-open"
+#define STOCK_LOCK_SECURE "stock_lock"
+#define STOCK_LOCK_BROKEN "stock_lock-broken"
+#define STOCK_NETWORK_OFFLINE "connect_no"
+#define STOCK_NETWORK_OFFLINE_ "network-offline"
+#define STOCK_SCRIPT "stock_script"
+#define STOCK_SEND "mail-send"
+#define STOCK_SEND_ "stock_mail-send"
+#define STOCK_TAB_NEW "tab-new"
+#define STOCK_TAB_NEW_ "stock_new-tab"
+#define STOCK_THEME "gnome-settings-theme"
+#define STOCK_USER_TRASH "user-trash"
+#define STOCK_USER_TRASH_ "gnome-stock-trash"
+#define STOCK_WINDOW_NEW "window-new"
+#define STOCK_WINDOW_NEW_ "stock_new-window"
+
+// For backwards compatibility
+
+#if !GTK_CHECK_VERSION(2, 10, 0)
+#define GTK_STOCK_SELECT_ALL "gtk-select-all"
+#endif
+
+#endif /* !__GLOBAL_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "helpers.h"
+
+#include "search.h"
+#include "sokoke.h"
+
+#include <string.h>
+#include <webkit.h>
+
+GtkIconTheme* get_icon_theme(GtkWidget* widget)
+{
+ return gtk_icon_theme_get_for_screen(gtk_widget_get_screen(widget));
+}
+
+GtkWidget* menu_item_new(const gchar* text, const gchar* icon
+ , GCallback signal, gboolean sensitive, gpointer userdata)
+{
+ GtkWidget* menuitem;
+ if(text)
+ menuitem = gtk_image_menu_item_new_with_mnemonic(text);
+ else
+ menuitem = gtk_image_menu_item_new_from_stock(icon, NULL);
+ if(icon)
+ {
+ GtkWidget* image = gtk_image_new_from_stock(icon, GTK_ICON_SIZE_MENU);
+ if(gtk_image_get_storage_type(GTK_IMAGE(image)) == GTK_IMAGE_EMPTY)
+ image = gtk_image_new_from_icon_name(icon, GTK_ICON_SIZE_MENU);
+ if(gtk_image_get_storage_type(GTK_IMAGE(image)) != GTK_IMAGE_EMPTY)
+ gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem), image);
+ else
+ g_print("Note: The icon %s is not available.", icon);
+ }
+ if(signal)
+ g_signal_connect(menuitem, "activate", signal, userdata);
+ gtk_widget_set_sensitive(GTK_WIDGET(menuitem), sensitive && signal);
+ return menuitem;
+}
+
+GtkToolItem* tool_button_new(const gchar* text, const gchar* icon
+ , gboolean important, gboolean sensitive, GCallback signal
+ , const gchar* tooltip, gpointer userdata)
+{
+ GtkToolItem* toolbutton = gtk_tool_button_new(NULL, NULL);
+ GtkStockItem stockItem;
+ if(gtk_stock_lookup(icon, &stockItem))
+ toolbutton = gtk_tool_button_new_from_stock(icon);
+ else
+ {
+ GtkIconTheme* iconTheme = get_icon_theme(GTK_WIDGET(toolbutton));
+ if(gtk_icon_theme_has_icon(iconTheme, icon))
+ gtk_tool_button_set_icon_name(GTK_TOOL_BUTTON(toolbutton), icon);
+ else
+ gtk_tool_button_set_stock_id(GTK_TOOL_BUTTON(toolbutton), GTK_STOCK_MISSING_IMAGE);
+ }
+ if(text)
+ gtk_tool_button_set_label(GTK_TOOL_BUTTON(toolbutton), text);
+ if(important)
+ gtk_tool_item_set_is_important(toolbutton, TRUE);
+ if(signal)
+ g_signal_connect(toolbutton, "clicked", signal, userdata);
+ gtk_widget_set_sensitive(GTK_WIDGET(toolbutton), sensitive && signal);
+ if(tooltip)
+ sokoke_tool_item_set_tooltip_text(toolbutton, tooltip);
+ return toolbutton;
+}
+
+GtkWidget* check_menu_item_new(const gchar* text
+ , GCallback signal, gboolean sensitive, gboolean active, CBrowser* browser)
+{
+ GtkWidget* menuitem = gtk_check_menu_item_new_with_mnemonic(text);
+ gtk_widget_set_sensitive(menuitem, sensitive && signal);
+ gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menuitem), active);
+ if(signal)
+ g_signal_connect(menuitem, "activate", signal, browser);
+ return menuitem;
+}
+
+GtkWidget* radio_button_new(GtkRadioButton* radio_button, const gchar* label)
+{
+ return gtk_radio_button_new_with_mnemonic_from_widget(radio_button, label);
+}
+
+void show_error(const gchar* text, const gchar* text2, CBrowser* browser)
+{
+ GtkWidget* dialog = gtk_message_dialog_new(
+ browser ? GTK_WINDOW(browser->window) : NULL
+ , 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_OK, text);
+ if(text2)
+ gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog), text2);
+ gtk_dialog_run(GTK_DIALOG(dialog));
+ gtk_widget_destroy(dialog);
+}
+
+gboolean spawn_protocol_command(const gchar* protocol, const gchar* res)
+{
+ const gchar* command = g_datalist_get_data(&config->protocols_commands, protocol);
+ if(!command)
+ return FALSE;
+
+ // Create an argument vector
+ gchar* uriEscaped = g_shell_quote(res);
+ gchar* commandReady;
+ if(strstr(command, "%s"))
+ commandReady = g_strdup_printf(command, uriEscaped);
+ else
+ commandReady = g_strconcat(command, " ", uriEscaped, NULL);
+ gchar** argv; GError* error = NULL;
+ if(!g_shell_parse_argv(commandReady, NULL, &argv, &error))
+ {
+ // FIXME: Should we have a more specific message?
+ show_error("Could not run external program.", error->message, NULL);
+ g_error_free(error);
+ g_free(commandReady); g_free(uriEscaped);
+ return FALSE;
+ }
+
+ // Try to run the command
+ error = NULL;
+ gboolean success = g_spawn_async(NULL, argv, NULL
+ , (GSpawnFlags)G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD
+ , NULL, NULL, NULL, &error);
+ g_strfreev(argv);
+
+ if(!success)
+ {
+ // FIXME: Should we have a more specific message?
+ show_error("Could not run external program.", error->message, NULL);
+ g_error_free(error);
+ }
+ g_free(commandReady); g_free(uriEscaped);
+ return TRUE;
+}
+
+GdkPixbuf* load_web_icon(const gchar* icon, GtkIconSize size, GtkWidget* widget)
+{
+ g_return_val_if_fail(GTK_IS_WIDGET(widget), NULL);
+ GdkPixbuf* pixbuf = NULL;
+ if(icon && *icon)
+ {
+ // TODO: We want to allow http as well, maybe also base64?
+ const gchar* iconReady = g_str_has_prefix(icon, "file://") ? &icon[7] : icon;
+ GtkStockItem stockItem;
+ if(gtk_stock_lookup(icon, &stockItem))
+ pixbuf = gtk_widget_render_icon(widget, iconReady, size, NULL);
+ else
+ {
+ gint width; gint height;
+ gtk_icon_size_lookup(size, &width, &height);
+ pixbuf = gtk_icon_theme_load_icon(gtk_icon_theme_get_default()
+ , icon, MAX(width, height), GTK_ICON_LOOKUP_USE_BUILTIN, NULL);
+ }
+ if(!pixbuf)
+ pixbuf = gdk_pixbuf_new_from_file_at_size(iconReady, 16, 16, NULL);
+ }
+ if(!pixbuf)
+ pixbuf = gtk_widget_render_icon(widget, GTK_STOCK_FIND, size, NULL);
+ return pixbuf;
+}
+
+void entry_setup_completion(GtkEntry* entry)
+{
+ /* TODO: The current behavior works only with the beginning of strings
+ But we want to match "localhost" with "loc" and "hos" */
+ GtkEntryCompletion* completion = gtk_entry_completion_new();
+ gtk_entry_completion_set_model(completion
+ , GTK_TREE_MODEL(gtk_list_store_new(1, G_TYPE_STRING)));
+ gtk_entry_completion_set_text_column(completion, 0);
+ gtk_entry_completion_set_minimum_key_length(completion, 3);
+ gtk_entry_set_completion(entry, completion);
+ gtk_entry_completion_set_popup_completion(completion, FALSE); //...
+}
+
+void entry_completion_append(GtkEntry* entry, const gchar* text)
+{
+ GtkEntryCompletion* completion = gtk_entry_get_completion(entry);
+ GtkTreeModel* completion_store = gtk_entry_completion_get_model(completion);
+ GtkTreeIter iter;
+ gtk_list_store_insert(GTK_LIST_STORE(completion_store), &iter, 0);
+ gtk_list_store_set(GTK_LIST_STORE(completion_store), &iter, 0, text, -1);
+}
+
+GtkWidget* get_nth_webView(gint n, CBrowser* browser)
+{
+ if(n < 0)
+ n = gtk_notebook_get_current_page(GTK_NOTEBOOK(browser->webViews));
+ GtkWidget* scrolled = gtk_notebook_get_nth_page(GTK_NOTEBOOK(browser->webViews), n);
+ return gtk_bin_get_child(GTK_BIN(scrolled));
+}
+
+gint get_webView_index(GtkWidget* webView, CBrowser* browser)
+{
+ GtkWidget* scrolled = gtk_widget_get_parent(webView);
+ return gtk_notebook_page_num(GTK_NOTEBOOK(browser->webViews), scrolled);
+}
+
+CBrowser* get_browser_from_webView(GtkWidget* webView)
+{
+ // FIXME: g_list_first
+ CBrowser* browser = NULL; GList* item = g_list_first(browsers);
+ do
+ {
+ browser = (CBrowser*)item->data;
+ if(browser->webView == webView)
+ return browser;
+ }
+ while((item = g_list_next(item)));
+ return NULL;
+}
+
+void update_favicon(CBrowser* browser)
+{
+ if(browser->loadedPercent == -1)
+ {
+ if(0) //browser->favicon // Has favicon?
+ {
+ // TODO: use custom icon
+ // gtk_image_set_from_file(GTK_IMAGE(browser->icon_page), "image");
+ }
+ else if(0) // Known mime-type?
+ {
+ // TODO: Retrieve mime type and load icon; don't forget ftp listings
+ }
+ else
+ gtk_image_set_from_stock(GTK_IMAGE(browser->webView_icon)
+ , GTK_STOCK_FILE, GTK_ICON_SIZE_MENU);
+ }
+ else
+ {
+ gtk_image_set_from_stock(GTK_IMAGE(browser->webView_icon)
+ , GTK_STOCK_EXECUTE, GTK_ICON_SIZE_MENU);
+ }
+}
+
+void update_security(CBrowser* browser)
+{
+ const gchar* uri = xbel_bookmark_get_href(browser->sessionItem);
+ // TODO: This check is bogus, until webkit tells us how secure a page is
+ if(g_str_has_prefix(uri, "https://"))
+ {
+ // TODO: highlighted entry indicates security, find an alternative
+ gtk_widget_modify_base(browser->location, GTK_STATE_NORMAL
+ , &browser->location->style->base[GTK_STATE_SELECTED]);
+ gtk_widget_modify_text(browser->location, GTK_STATE_NORMAL
+ , &browser->location->style->text[GTK_STATE_SELECTED]);
+ gtk_image_set_from_stock(GTK_IMAGE(browser->icon_security)
+ , GTK_STOCK_DIALOG_AUTHENTICATION, GTK_ICON_SIZE_MENU);
+ }
+ else
+ {
+ gtk_widget_modify_base(browser->location, GTK_STATE_NORMAL, NULL);
+ gtk_widget_modify_text(browser->location, GTK_STATE_NORMAL, NULL);
+ gtk_image_set_from_stock(GTK_IMAGE(browser->icon_security)
+ , GTK_STOCK_INFO, GTK_ICON_SIZE_MENU);
+ }
+}
+
+void update_visibility(CBrowser* browser, gboolean visibility)
+{
+ // A tabbed window shouldn't be manipulatable
+ if(gtk_notebook_get_n_pages(GTK_NOTEBOOK(browser->webViews)) > 1)
+ return;
+
+ // SHOULD SCRIPTS BE ABLE TO HIDE WINDOWS AT ALL?
+ if(0 && !visibility)
+ {
+ gtk_widget_hide(browser->window);
+ return;
+ }
+ else if(!visibility)
+ g_print("Window was not hidden.\n");
+
+ sokoke_widget_set_visible(browser->menubar, browser->hasMenubar);
+ sokoke_widget_set_visible(browser->navibar, browser->hasToolbar);
+ sokoke_widget_set_visible(browser->location, browser->hasLocation);
+ sokoke_widget_set_visible(browser->webSearch, browser->hasLocation);
+ sokoke_widget_set_visible(browser->statusbar, browser->hasStatusbar);
+}
+
+void action_set_active(const gchar* name, gboolean active, CBrowser* browser)
+{
+ // This shortcut toggles activity state by an action name
+ GtkAction* action = gtk_action_group_get_action(browser->actiongroup, name);
+ g_return_if_fail(GTK_IS_ACTION(action));
+ gtk_toggle_action_set_active(GTK_TOGGLE_ACTION(action), active);
+}
+
+void action_set_sensitive(const gchar* name, gboolean sensitive, CBrowser* browser)
+{
+ // This shortcut toggles sensitivity by an action name
+ GtkAction* action = gtk_action_group_get_action(browser->actiongroup, name);
+ g_return_if_fail(GTK_IS_ACTION(action));
+ gtk_action_set_sensitive(action, sensitive);
+}
+
+void action_set_visible(const gchar* name, gboolean visible, CBrowser* browser)
+{
+ // This shortcut toggles visibility by an action name
+ GtkAction* action = gtk_action_group_get_action(browser->actiongroup, name);
+ g_return_if_fail(GTK_IS_ACTION(action));
+ gtk_action_set_visible(action, visible);
+}
+
+void update_statusbar_text(CBrowser* browser)
+{
+ if(browser->statusMessage)
+ {
+ gtk_statusbar_pop(GTK_STATUSBAR(browser->statusbar), 1);
+ gtk_statusbar_push(GTK_STATUSBAR(browser->statusbar), 1
+ , browser->statusMessage);
+ }
+ else
+ {
+ gchar* message;
+ if(browser->loadedPercent)
+ message = g_strdup_printf("%d%% loaded, %d/%d bytes"
+ , browser->loadedPercent, browser->loadedBytes, browser->loadedBytesMax);
+ else if(browser->loadedBytes)
+ message = g_strdup_printf("%d bytes", browser->loadedBytes);
+ else
+ message = g_strdup(" ");
+ gtk_progress_bar_set_text(GTK_PROGRESS_BAR(browser->progress), message);
+ g_free(message);
+ }
+}
+
+void update_edit_items(CBrowser* browser)
+{
+ GtkWidget* widget = gtk_window_get_focus(GTK_WINDOW(browser->window));
+ gboolean hasSelection = FALSE;
+ gboolean canCut = FALSE; gboolean canCopy = FALSE; gboolean canPaste = FALSE;
+ if(widget && (/*WEBKIT_IS_WEB_VIEW(widget) || */GTK_IS_EDITABLE(widget)))
+ {
+ hasSelection = /*WEBKIT_IS_WEB_VIEW(widget)
+ ? webkit_web_view_has_selection(WEBKIT_WEB_VIEW(widget), NULL, NULL)
+ : */gtk_editable_get_selection_bounds(GTK_EDITABLE(widget), NULL, NULL);
+ canCut = /*WEBKIT_IS_WEB_VIEW(widget)
+ ? webkit_web_view_can_cut_clipboard(WEBKIT_WEB_VIEW(widget))
+ : */hasSelection && gtk_editable_get_editable(GTK_EDITABLE(widget));
+ canCopy = /*WEBKIT_IS_WEB_VIEW(widget)
+ ? webkit_web_view_can_copy_clipboard(WEBKIT_WEB_VIEW(widget))
+ : */hasSelection;
+ canPaste = /*WEBKIT_IS_WEB_VIEW(widget)
+ ? webkit_web_view_can_paste_clipboard(WEBKIT_WEB_VIEW(widget))
+ : */gtk_editable_get_editable(GTK_EDITABLE(widget));
+ }
+ action_set_sensitive("Cut", canCut, browser);
+ action_set_sensitive("Copy", canCopy, browser);
+ action_set_sensitive("Paste", canPaste, browser);
+ action_set_sensitive("Delete", canCut, browser);
+ action_set_sensitive("SelectAll", !hasSelection, browser);
+}
+
+void update_gui_state(CBrowser* browser)
+{
+ GtkWidget* webView = get_nth_webView(-1, browser);
+ action_set_sensitive("ZoomIn", FALSE, browser);//webkit_web_view_can_increase_text_size(WEBKIT_WEB_VIEW(webView), browser);
+ action_set_sensitive("ZoomOut", FALSE, browser);//webkit_web_view_can_decrease_text_size(WEBKIT_WEB_VIEW(webView)), browser);
+ action_set_sensitive("ZoomNormal", FALSE, browser);//webkit_web_view_get_text_size(WEBKIT_WEB_VIEW(webView)) != 1, browser);
+ action_set_sensitive("Back", webkit_web_view_can_go_backward(WEBKIT_WEB_VIEW(webView)), browser);
+ action_set_sensitive("Forward", webkit_web_view_can_go_forward(WEBKIT_WEB_VIEW(webView)), browser);
+ action_set_sensitive("Refresh", browser->loadedPercent == -1, browser);
+ action_set_sensitive("Stop", browser->loadedPercent != -1, browser);
+
+ GtkAction* action = gtk_action_group_get_action(browser->actiongroup, "RefreshStop");
+ if(browser->loadedPercent == -1)
+ {
+ gtk_widget_hide(browser->throbber);
+ g_object_set(action, "stock-id", GTK_STOCK_REFRESH, NULL);
+ g_object_set(action, "tooltip", "Refresh the current page", NULL);
+ }
+ else
+ {
+ gtk_widget_show(browser->throbber);
+ g_object_set(action, "stock-id", GTK_STOCK_STOP, NULL);
+ g_object_set(action, "tooltip", "Stop loading the current page", NULL);
+ }
+
+ gtk_image_set_from_stock(GTK_IMAGE(browser->location_icon), GTK_STOCK_FILE
+ , GTK_ICON_SIZE_MENU);
+
+ if(browser->loadedBytesMax < 1) // Skip null and negative values
+ {
+ gtk_progress_bar_pulse(GTK_PROGRESS_BAR(browser->progress));
+ update_statusbar_text(browser);
+ }
+ else
+ {
+ if(browser->loadedBytes > browser->loadedBytesMax)
+ browser->loadedPercent = 100;
+ else
+ browser->loadedPercent
+ = (browser->loadedBytes * 100) / browser->loadedBytesMax;
+ gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(browser->progress)
+ , browser->loadedPercent / 100.0);
+ update_statusbar_text(browser);
+ }
+}
+
+void update_feeds(CBrowser* browser)
+{
+ // TODO: Look for available feeds, requires dom access
+}
+
+void update_search_engines(CBrowser* browser)
+{
+ // TODO: Look for available search engines, requires dom access
+}
+
+void update_status_message(const gchar* message, CBrowser* browser)
+{
+ g_free(browser->statusMessage);
+ browser->statusMessage = g_strdup(message ? message : "");
+ update_statusbar_text(browser);
+}
+
+void update_browser_actions(CBrowser* browser)
+{
+ gboolean active = gtk_notebook_get_n_pages(GTK_NOTEBOOK(browser->webViews)) > 1;
+ gtk_notebook_set_show_tabs(GTK_NOTEBOOK(browser->webViews), active);
+ action_set_sensitive("TabClose", active, browser);
+ guint n = xbel_folder_get_n_items(tabtrash);
+ action_set_sensitive("UndoTabClose", n, browser);
+ action_set_sensitive("TabsClosed", n, browser);
+}
+
+gchar* magic_uri(const gchar* uri, gboolean search)
+{
+ // Add file:// if we have a local path
+ if(g_path_is_absolute(uri))
+ return g_strconcat("file://", uri, NULL);
+ // Do we need to add a protocol?
+ if(!strstr(uri, "://"))
+ {
+ // Do we have a domain, ip address or localhost?
+ if(strstr(uri, ".") != NULL || !strcmp(uri, "localhost"))
+ return g_strconcat("http://", uri, NULL);
+ // We don't want to search? So return early.
+ if(!search)
+ return g_strdup(uri);
+ gchar search[256];
+ const gchar* searchUrl = NULL;
+ // Do we have a keyword and a string?
+ gchar** parts = g_strsplit(uri, " ", 2);
+ if(parts[0] && parts[1])
+ {
+ guint n = g_list_length(searchEngines);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ SearchEngine* searchEngine = (SearchEngine*)g_list_nth_data(searchEngines, i);
+ if(!strcmp(search_engine_get_keyword(searchEngine), parts[0]))
+ searchUrl = searchEngine->url;
+ }
+ if(searchUrl != NULL)
+ g_snprintf(search, 255, searchUrl, parts[1]);
+ }
+ //g_strfreev(sParts);
+ // We only have a word or there is no matching keyowrd, so search for it
+ if(searchUrl == NULL)
+ g_snprintf(search, 255, config->locationSearch, uri);
+ return g_strdup(search);
+ }
+ return g_strdup(uri);
+}
+
+gchar* get_default_font(void)
+{
+ GtkSettings* gtksettings = gtk_settings_get_default();
+ gchar* defaultFont;
+ g_object_get(gtksettings, "gtk-font-name", &defaultFont, NULL);
+ return defaultFont;
+}
+
+GtkToolbarStyle config_to_toolbarstyle(guint toolbarStyle)
+{
+ switch(toolbarStyle)
+ {
+ case CONFIG_TOOLBAR_ICONS:
+ return GTK_TOOLBAR_ICONS;
+ case CONFIG_TOOLBAR_TEXT:
+ return GTK_TOOLBAR_TEXT;
+ case CONFIG_TOOLBAR_BOTH:
+ return GTK_TOOLBAR_BOTH;
+ case CONFIG_TOOLBAR_BOTH_HORIZ:
+ return GTK_TOOLBAR_BOTH_HORIZ;
+ }
+ GtkSettings* gtkSettings = gtk_settings_get_default();
+ g_object_get(gtkSettings, "gtk-toolbar-style", &toolbarStyle, NULL);
+ return toolbarStyle;
+}
+
+GtkToolbarStyle config_to_toolbariconsize(gboolean toolbarSmall)
+{
+ return toolbarSmall ? GTK_ICON_SIZE_SMALL_TOOLBAR
+ : GTK_ICON_SIZE_LARGE_TOOLBAR;
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __HELPERS_H__
+#define __HELPERS_H__ 1
+
+#include "browser.h"
+
+#include <gtk/gtk.h>
+
+GtkIconTheme*
+get_icon_theme(GtkWidget*);
+
+GtkWidget*
+menu_item_new(const gchar*, const gchar*, GCallback, gboolean, gpointer);
+
+GtkToolItem*
+tool_button_new(const gchar*, const gchar*
+ , gboolean, gboolean, GCallback, const gchar*, gpointer);
+
+GtkWidget*
+check_menu_item_new(const gchar*, GCallback, gboolean, gboolean, CBrowser*);
+
+GtkWidget*
+radio_button_new(GtkRadioButton*, const gchar*);
+
+void
+show_error(const gchar*, const gchar*, CBrowser*);
+
+gboolean
+spawn_protocol_command(const gchar*, const gchar*);
+
+GdkPixbuf*
+load_web_icon(const gchar*, GtkIconSize, GtkWidget*);
+
+void
+entry_setup_completion(GtkEntry*);
+
+void
+entry_completion_append(GtkEntry*, const gchar*);
+
+GtkWidget*
+get_nth_webView(gint, CBrowser*);
+
+gint
+get_webView_index(GtkWidget*, CBrowser*);
+
+CBrowser*
+get_browser_from_webView(GtkWidget*);
+
+void
+update_favicon(CBrowser*);
+
+void
+update_security(CBrowser*);
+
+void
+update_visibility(CBrowser*, gboolean);
+
+void
+action_set_active(const gchar*, gboolean, CBrowser*);
+
+void
+action_set_sensitive(const gchar*, gboolean, CBrowser*);
+
+void
+action_set_visible(const gchar*, gboolean, CBrowser*);
+
+void
+update_statusbar_text(CBrowser*);
+
+void
+update_edit_items(CBrowser*);
+
+void
+update_gui_state(CBrowser*);
+
+void
+update_feeds(CBrowser*);
+
+void
+update_search_engines(CBrowser*);
+
+void
+update_status_message(const gchar*, CBrowser*);
+
+void
+update_browser_actions(CBrowser*);
+
+gchar*
+magic_uri(const gchar*, gboolean bSearch);
+
+gchar*
+get_default_font(void);
+
+GtkToolbarStyle
+config_to_toolbarstyle();
+
+GtkToolbarStyle
+config_to_toolbariconsize(gboolean);
+
+#endif /* !__HELPERS_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "main.h"
+
+#include "browser.h"
+#include "global.h"
+#include "helpers.h"
+#include "sokoke.h"
+#include "search.h"
+#include "webView.h"
+#include "xbel.h"
+
+#include <string.h>
+#include <gtk/gtk.h>
+
+#include "config.h"
+
+// -- stock icons
+
+static void stock_items_init(void)
+{
+ static GtkStockItem items[] =
+ {
+ { STOCK_LOCK_OPEN },
+ { STOCK_LOCK_SECURE },
+ { STOCK_LOCK_BROKEN },
+ { STOCK_SCRIPT },
+ { STOCK_THEME },
+ { STOCK_USER_TRASH },
+
+ { STOCK_BOOKMARK, "Bookmark", 0, 0, NULL },
+ { STOCK_BOOKMARK_NEW, "New Bookmark", 0, 0, NULL },
+ { STOCK_BOOKMARKS, "_Bookmarks", 0, 0, NULL },
+ { STOCK_DOWNLOADS, "_Downloads", 0, 0, NULL },
+ { STOCK_CONSOLE, "_Console", 0, 0, NULL },
+ { STOCK_EXTENSIONS, "_Extensions", 0, 0, NULL },
+ { STOCK_FORM_FILL, "_Form Fill", 0, 0, NULL },
+ { STOCK_HISTORY, "History", 0, 0, NULL },
+ { STOCK_HOMEPAGE, "Homepage", 0, 0, NULL },
+ { STOCK_LOCATION, "Location Entry", 0, 0, NULL },
+ { STOCK_NEWSFEED, "Newsfeed", 0, 0, NULL },
+ { STOCK_PLUGINS, "_Plugins", 0, 0, NULL },
+ { STOCK_POPUPS_BLOCKED, "Blocked Popups", 0, 0, NULL },
+ { STOCK_SOURCE_VIEW, "View Source", 0, 0, NULL },
+ { STOCK_TAB_CLOSE, "C_lose Tab", 0, 0, NULL },
+ { STOCK_TAB_NEW, "New _Tab", 0, 0, NULL },
+ { STOCK_WINDOW_CLOSE, "_Close Window", 0, 0, NULL },
+ { STOCK_WINDOW_NEW, "New _Window", 0, 0, NULL },
+ #if !GTK_CHECK_VERSION(2, 10, 0)
+ { GTK_STOCK_SELECT_ALL, "Select _All", 0, 0, (gchar*)"gtk20" },
+ #endif
+ };
+ GtkIconFactory* factory = gtk_icon_factory_new();
+ guint i;
+ for(i = 0; i < (guint)G_N_ELEMENTS(items); i++)
+ {
+ GtkIconSource* iconSource = gtk_icon_source_new();
+ gtk_icon_source_set_icon_name(iconSource, items[i].stock_id);
+ GtkIconSet* iconSet = gtk_icon_set_new();
+ gtk_icon_set_add_source(iconSet, iconSource);
+ gtk_icon_source_free(iconSource);
+ gtk_icon_factory_add(factory, items[i].stock_id, iconSet);
+ gtk_icon_set_unref(iconSet);
+ }
+ gtk_stock_add_static(items, G_N_ELEMENTS(items));
+ gtk_icon_factory_add_default(factory);
+ g_object_unref(factory);
+}
+
+// -- main function
+
+int main(int argc, char** argv)
+{
+ g_set_application_name(PACKAGE_NAME);
+
+ // Parse cli options
+ gint repeats = 2;
+ gboolean version = FALSE;
+ GOptionEntry entries[] =
+ {
+ { "repeats", 'r', 0, G_OPTION_ARG_INT, &repeats, "An unused value", "N" },
+ { "version", 'v', 0, G_OPTION_ARG_NONE, &version, "Display program version", NULL }
+ };
+
+ GError* error = NULL;
+ if(!gtk_init_with_args(&argc, &argv, "[URI]", entries, NULL/*GETTEXT_PACKAGE*/, &error))
+ {
+ g_error_free(error);
+ return 1;
+ }
+
+ if(version)
+ {
+ g_print(PACKAGE_STRING " - Copyright (c) 2007 Christian Dywan\n\n"
+ "GTK+2: " GTK_VER "\n"
+ "WebKit: " WEBKIT_VER "\n"
+ "Libsexy: " LIBSEXY_VER "\n"
+ "libXML2: " LIBXML_VER "\n"
+ "GetText: N/A\n"
+ "\n"
+ "Debugging: " SOKOKE_DEBUG_ "\n"
+ "\n"
+ "Please report comments, suggestions and bugs to:\n"
+ "\t" PACKAGE_BUGREPORT "\n"
+ "Check for new versions at:\n"
+ "\thttp://software.twotoasts.de\n");
+ return 0;
+ }
+
+ // Load configuration files
+ GString* errorMessages = g_string_new(NULL);
+ // TODO: What about default config in a global config folder?
+ gchar* configPath = g_build_filename(g_get_user_config_dir(), PACKAGE_NAME, NULL);
+ g_mkdir_with_parents(configPath, 0755);
+ gchar* configFile = g_build_filename(configPath, "config", NULL);
+ error = NULL;
+ config = config_new();
+ if(!config_from_file(config, configFile, &error))
+ {
+ if(error->code != G_FILE_ERROR_NOENT)
+ g_string_append_printf(errorMessages
+ , "Configuration was not loaded. %s\n", error->message);
+ g_error_free(error);
+ }
+ g_free(configFile);
+ configFile = g_build_filename(configPath, "accels", NULL);
+ gtk_accel_map_load(configFile);
+ g_free(configFile);
+ configFile = g_build_filename(configPath, "search", NULL);
+ error = NULL;
+ searchEngines = search_engines_new();
+ if(!search_engines_from_file(&searchEngines, configFile, &error))
+ {
+ // FIXME: We may have a "file empty" error, how do we recognize that?
+ /*if(error->code != G_FILE_ERROR_NOENT)
+ g_string_append_printf(errorMessages
+ , "Notice: No search engines loaded. %s\n", error->message);*/
+ g_error_free(error);
+ }
+ g_free(configFile);
+ configFile = g_build_filename(configPath, "bookmarks.xbel", NULL);
+ bookmarks = xbel_folder_new();
+ error = NULL;
+ if(!xbel_folder_from_file(bookmarks, configFile, &error))
+ {
+ if(error->code != G_FILE_ERROR_NOENT)
+ g_string_append_printf(errorMessages
+ , "Bookmarks couldn't be loaded. %s\n", error->message);
+ g_error_free(error);
+ }
+ g_free(configFile);
+ XbelItem* _session = xbel_folder_new();
+ if(config->startup == CONFIG_STARTUP_SESSION)
+ {
+ configFile = g_build_filename(configPath, "session.xbel", NULL);
+ error = NULL;
+ if(!xbel_folder_from_file(_session, configFile, &error))
+ {
+ if(error->code != G_FILE_ERROR_NOENT)
+ g_string_append_printf(errorMessages
+ , "Session couldn't be loaded. %s\n", error->message);
+ g_error_free(error);
+ }
+ g_free(configFile);
+ }
+ configFile = g_build_filename(configPath, "tabtrash.xbel", NULL);
+ tabtrash = xbel_folder_new();
+ error = NULL;
+ if(!xbel_folder_from_file(tabtrash, configFile, &error))
+ {
+ if(error->code != G_FILE_ERROR_NOENT)
+ g_string_append_printf(errorMessages
+ , "Tabtrash couldn't be loaded. %s\n", error->message);
+ g_error_free(error);
+ }
+ g_free(configFile);
+
+ // In case of errors
+ if(errorMessages->len)
+ {
+ GtkWidget* dialog = gtk_message_dialog_new(NULL
+ , 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_NONE
+ , "The following errors occured.");
+ gtk_window_set_skip_taskbar_hint(GTK_WINDOW(dialog), FALSE);
+ gtk_window_set_title(GTK_WINDOW(dialog), g_get_application_name());
+ // FIXME: Use custom program icon
+ gtk_window_set_icon_name(GTK_WINDOW(dialog), "web-browser");
+ gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog)
+ , "%s", errorMessages->str);
+ gtk_dialog_add_buttons(GTK_DIALOG(dialog)
+ , GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL
+ , "_Ignore", GTK_RESPONSE_ACCEPT
+ , NULL);
+ if(gtk_dialog_run(GTK_DIALOG(dialog)) != GTK_RESPONSE_ACCEPT)
+ {
+ config_free(config);
+ search_engines_free(searchEngines);
+ xbel_item_free(bookmarks);
+ xbel_item_free(_session);
+ g_string_free(errorMessages, TRUE);
+ return 0;
+ }
+ gtk_widget_destroy(dialog);
+ /* FIXME: Since we will overwrite files that could not be loaded
+ , would we want to make backups? */
+ }
+ g_string_free(errorMessages, TRUE);
+
+ if(xbel_folder_is_empty(_session))
+ {
+ XbelItem* item = xbel_bookmark_new();
+ if(config->startup == CONFIG_STARTUP_HOMEPAGE)
+ xbel_bookmark_set_href(item, config->homepage);
+ else
+ xbel_bookmark_set_href(item, "about:blank");
+ xbel_folder_append_item(_session, item);
+ }
+ g_free(configPath);
+
+ accel_group = gtk_accel_group_new();
+ stock_items_init();
+ browsers = NULL;
+
+ // TODO: Handle any number of separate uris from argv
+ // Open as many tabs as we have uris, seperated by pipes
+ gchar* uri = argc > 1 ? strtok(g_strdup(argv[1]), "|") : NULL;
+ while(uri != NULL)
+ {
+ XbelItem* item = xbel_bookmark_new();
+ gchar* uriReady = magic_uri(uri, FALSE);
+ xbel_bookmark_set_href(item, uriReady);
+ g_free(uriReady);
+ xbel_folder_append_item(_session, item);
+ uri = strtok(NULL, "|");
+ }
+ g_free(uri);
+
+ session = xbel_folder_new();
+ CBrowser* browser = NULL;
+ guint n = xbel_folder_get_n_items(_session);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ XbelItem* item = xbel_folder_get_nth_item(_session, i);
+ browser = browser_new(browser);
+ webView_open(browser->webView, xbel_bookmark_get_href(item));
+ }
+ xbel_item_free(_session);
+
+ gtk_main();
+
+ // Save configuration files
+ configPath = g_build_filename(g_get_user_config_dir(), PACKAGE_NAME, NULL);
+ g_mkdir_with_parents(configPath, 0755);
+ configFile = g_build_filename(configPath, "search", NULL);
+ error = NULL;
+ if(!search_engines_to_file(searchEngines, configFile, &error))
+ {
+ g_warning("Search engines couldn't be saved. %s", error->message);
+ g_error_free(error);
+ }
+ search_engines_free(searchEngines);
+ g_free(configFile);
+ configFile = g_build_filename(configPath, "bookmarks.xbel", NULL);
+ error = NULL;
+ if(!xbel_folder_to_file(bookmarks, configFile, &error))
+ {
+ g_warning("Bookmarks couldn't be saved. %s", error->message);
+ g_error_free(error);
+ }
+ xbel_item_free(bookmarks);
+ g_free(configFile);
+ configFile = g_build_filename(configPath, "tabtrash.xbel", NULL);
+ error = NULL;
+ if(!xbel_folder_to_file(tabtrash, configFile, &error))
+ {
+ g_warning("Tabtrash couldn't be saved. %s", error->message);
+ g_error_free(error);
+ }
+ xbel_item_free(tabtrash);
+ g_free(configFile);
+ if(config->startup == CONFIG_STARTUP_SESSION)
+ {
+ configFile = g_build_filename(configPath, "session.xbel", NULL);
+ error = NULL;
+ if(!xbel_folder_to_file(session, configFile, &error))
+ {
+ g_warning("Session couldn't be saved. %s", error->message);
+ g_error_free(error);
+ }
+ g_free(configFile);
+ }
+ xbel_item_free(session);
+ configFile = g_build_filename(configPath, "config", NULL);
+ error = NULL;
+ if(!config_to_file(config, configFile, &error))
+ {
+ g_warning("Configuration couldn't be saved. %s", error->message);
+ g_error_free(error);
+ }
+ config_free(config);
+ g_free(configFile);
+ configFile = g_build_filename(configPath, "accels", NULL);
+ gtk_accel_map_save(configFile);
+ g_free(configFile);
+ g_free(configPath);
+ return 0;
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __MIDORI_H__
+#define __MIDORI_H__ 1
+
+#endif /* !__MIDORI_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "prefs.h"
+
+#include "helpers.h"
+#include "global.h"
+#include "sokoke.h"
+
+#include "string.h"
+
+static gboolean on_prefs_homepage_focus_out(GtkWidget* widget
+ , GdkEventFocus event, CPrefs* prefs)
+{
+ g_free(config->homepage);
+ config->homepage = g_strdup(gtk_entry_get_text(GTK_ENTRY(widget)));
+ return FALSE;
+}
+
+static void on_prefs_loadonstartup_changed(GtkWidget* widget, CPrefs* prefs)
+{
+ config->startup = gtk_combo_box_get_active(GTK_COMBO_BOX(widget));
+}
+
+static void on_prefs_newpages_changed(GtkWidget* widget, CPrefs* prefs)
+{
+ config->newPages = gtk_combo_box_get_active(GTK_COMBO_BOX(widget));
+}
+
+void on_prefs_openTabsInTheBackground_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->openTabsInTheBackground = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+}
+
+static void on_prefs_openPopupsInTabs_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->openPopupsInTabs = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+}
+
+static void on_prefs_loadImagesAutomatically_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->loadImagesAutomatically = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ // FIXME: Apply the change to all open webViews
+ g_object_set(get_nth_webView(-1, prefs->browser)
+ , "loads-images-automatically", config->loadImagesAutomatically, NULL);
+}
+
+static void on_prefs_shrinkImagesToFit_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->shrinkImagesToFit = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ // FIXME: Apply the change to all open webViews
+ g_object_set(get_nth_webView(-1, prefs->browser)
+ , "shrinks-standalone-images-to-fit", config->shrinkImagesToFit, NULL);
+}
+
+static void on_prefs_resizableTextAreas_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->resizableTextAreas = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ // FIXME: Apply the change to all open webViews
+ g_object_set(get_nth_webView(-1, prefs->browser)
+ , "text-areas-are-resizable", config->resizableTextAreas, NULL);
+}
+
+static void on_prefs_enableJavaScript_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->enableJavaScript = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ // FIXME: Apply the change to all open webViews
+ g_object_set(get_nth_webView(-1, prefs->browser)
+ , "java-script-enabled", config->enableJavaScript, NULL);
+}
+
+static void on_prefs_enablePlugins_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->enablePlugins = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ // FIXME: Apply the change to all open webViews
+ g_object_set(get_nth_webView(-1, prefs->browser)
+ , "plugins-enabled", config->enablePlugins, NULL);
+}
+
+static void on_prefs_toolbarstyle_changed(GtkWidget* widget, CPrefs* prefs)
+{
+ config->toolbarStyle = gtk_combo_box_get_active(GTK_COMBO_BOX(widget));
+ gtk_toolbar_set_style(GTK_TOOLBAR(prefs->browser->navibar)
+ , config_to_toolbarstyle(config->toolbarStyle));
+}
+
+static void on_prefs_toolbarSmall_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->toolbarSmall = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ gtk_toolbar_set_icon_size(GTK_TOOLBAR(prefs->browser->navibar)
+ , config_to_toolbariconsize(config->toolbarSmall));
+}
+
+static void on_prefs_closeButtonsOnTabs_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->tabClose = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ GList* items = browsers;
+ do
+ {
+ CBrowser* browser = (CBrowser*)items->data;
+ sokoke_widget_set_visible(browser->webView_close, config->tabClose);
+ }
+ while((items = g_list_next(items)));
+ g_list_free(items);
+}
+
+static void on_prefs_toolbarWebSearch_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->toolbarWebSearch = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ sokoke_widget_set_visible(prefs->browser->webSearch, config->toolbarWebSearch);
+}
+
+static void on_prefs_toolbarNewTab_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->toolbarNewTab = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ sokoke_widget_set_visible(prefs->browser->newTab, config->toolbarNewTab);
+}
+
+static void on_prefs_toolbarClosedTabs_toggled(GtkWidget* widget, CPrefs* prefs)
+{
+ config->toolbarClosedTabs = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ sokoke_widget_set_visible(prefs->browser->closedTabs, config->toolbarClosedTabs);
+}
+
+static gboolean on_prefs_locationsearch_focus_out(GtkWidget* widget
+ , GdkEventFocus event, CPrefs* prefs)
+{
+ g_free(config->locationSearch);
+ config->locationSearch = g_strdup(gtk_entry_get_text(GTK_ENTRY(widget)));
+ return FALSE;
+}
+
+static void on_prefs_protocols_render_icon(GtkTreeViewColumn* column
+ , GtkCellRenderer* renderer, GtkTreeModel* model, GtkTreeIter* iter, CPrefs* prefs)
+{
+ gchar* command;
+ gtk_tree_model_get(model, iter, PROTOCOLS_COL_COMMAND, &command, -1);
+
+ // TODO: Would it be better, to not do this on every redraw?
+ // Determine the actual binary to be able to display an icon
+ gchar* binary = NULL;
+ if(command)
+ binary = strtok(command, " ");
+ if(binary)
+ {
+ gchar* path;
+ if((path = g_find_program_in_path(binary)))
+ {
+ GtkIconTheme* icon_theme = get_icon_theme(prefs->treeview);
+ if(g_path_is_absolute(binary))
+ {
+ g_free(path); path = g_path_get_basename(binary);
+ }
+ // TODO: Is it good to just display nothing if there is no icon?
+ if(!gtk_icon_theme_has_icon(icon_theme, binary))
+ binary = NULL;
+ #if GTK_CHECK_VERSION(2, 10, 0)
+ g_object_set(renderer, "icon-name", binary, NULL);
+ #else
+ GdkPixbuf* icon = binary != NULL
+ ? gtk_icon_theme_load_icon(gtk_icon_theme_get_default()
+ , binary, GTK_ICON_SIZE_MENU, 0, NULL) : NULL;
+ g_object_set(renderer, "pixbuf", icon, NULL);
+ if(icon)
+ g_object_unref(icon);
+ #endif
+ g_free(path);
+ }
+ else
+ {
+ #if GTK_CHECK_VERSION(2, 10, 0)
+ g_object_set(renderer, "icon-name", NULL, NULL);
+ #endif
+ g_object_set(renderer, "stock-id", GTK_STOCK_DIALOG_ERROR, NULL);
+ }
+ }
+ else
+ {
+ // We need to reset the icon
+ #if GTK_CHECK_VERSION(2, 10, 0)
+ g_object_set(renderer, "icon-name", NULL, NULL);
+ #else
+ g_object_set(renderer, "stock-id", NULL, NULL);
+ #endif
+ }
+ g_free(command);
+}
+
+static void on_prefs_protocols_edited(GtkCellRendererText* renderer
+ , gchar* path, gchar* textNew, CPrefs* prefs)
+{
+ GtkTreeModel* model = gtk_tree_view_get_model(GTK_TREE_VIEW(prefs->treeview));
+ GtkTreeIter iter;
+ gtk_tree_model_get_iter_from_string(model, &iter, path);
+ gtk_list_store_set(GTK_LIST_STORE(model), &iter
+ , PROTOCOLS_COL_COMMAND, textNew, -1);
+ gchar* protocol;
+ gtk_tree_model_get(model, &iter, PROTOCOLS_COL_NAME, &protocol, -1);
+ g_datalist_set_data_full(&config->protocols_commands
+ , protocol, g_strdup(textNew), g_free);
+}
+
+static void on_prefs_protocols_add_clicked(GtkWidget* widget, CPrefs* prefs)
+{
+ gchar* protocol = gtk_combo_box_get_active_text(GTK_COMBO_BOX(prefs->combobox));
+ GtkTreeModel* liststore = gtk_tree_view_get_model(GTK_TREE_VIEW(prefs->treeview));
+ gtk_list_store_insert_with_values(GTK_LIST_STORE(liststore), NULL, G_MAXINT
+ , PROTOCOLS_COL_NAME, protocol
+ , PROTOCOLS_COL_COMMAND, "", -1);
+ g_ptr_array_add(config->protocols_names, (gpointer)protocol);
+ g_datalist_set_data_full(&config->protocols_commands
+ , protocol, g_strdup(""), g_free);
+ gtk_widget_set_sensitive(prefs->add, FALSE);
+}
+
+static void on_prefs_protocols_combobox_changed(GtkWidget* widget, CPrefs* prefs)
+{
+ gchar* protocol = gtk_combo_box_get_active_text(GTK_COMBO_BOX(widget));
+ gchar* command = (gchar*)g_datalist_get_data(&config->protocols_commands, protocol);
+ g_free(protocol);
+ gtk_widget_set_sensitive(prefs->add, command == NULL);
+}
+
+GtkWidget* prefs_preferences_dialog_new(CBrowser* browser)
+{
+ gchar* dialogTitle = g_strdup_printf("%s Preferences", g_get_application_name());
+ GtkWidget* dialog = gtk_dialog_new_with_buttons(dialogTitle
+ , GTK_WINDOW(browser->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR
+ , GTK_STOCK_HELP
+ , GTK_RESPONSE_HELP
+ , GTK_STOCK_CLOSE
+ , GTK_RESPONSE_CLOSE
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog), GTK_STOCK_PREFERENCES);
+ // TODO: Implement some kind of help function
+ gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog), GTK_RESPONSE_HELP, FALSE); //...
+ g_signal_connect(dialog, "response", G_CALLBACK(gtk_widget_destroy), dialog);
+
+ CPrefs* prefs = g_new0(CPrefs, 1);
+ prefs->browser = browser;
+ //prefs->window = dialog;
+ g_signal_connect(dialog, "response", G_CALLBACK(g_free), prefs);
+
+ // TODO: Do we want tooltips for explainations or can we omit that?
+ // TODO: We need mnemonics
+ // TODO: Take multiple windows into account when applying changes
+ GtkWidget* xfce_heading;
+ if((xfce_heading = sokoke_xfce_header_new(
+ gtk_window_get_icon_name(GTK_WINDOW(browser->window)), dialogTitle)))
+ gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox)
+ , xfce_heading, FALSE, FALSE, 0);
+ g_free(dialogTitle);
+ GtkWidget* notebook = gtk_notebook_new();
+ gtk_container_set_border_width(GTK_CONTAINER(notebook), 6);
+ GtkSizeGroup* sizegroup = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
+ GtkSizeGroup* sizegroup2 = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
+ GtkWidget* page; GtkWidget* frame; GtkWidget* table; GtkWidget* align;
+ GtkWidget* button; GtkWidget* checkbutton; GtkWidget* colorbutton;
+ GtkWidget* combobox; GtkWidget* entry; GtkWidget* hbox; GtkWidget* spinbutton;
+ #define PAGE_NEW(__label) page = gtk_vbox_new(FALSE, 0);\
+ gtk_container_set_border_width(GTK_CONTAINER(page), 5);\
+ gtk_notebook_append_page(GTK_NOTEBOOK(notebook), page, gtk_label_new(__label))
+ #define FRAME_NEW(__label) frame = sokoke_hig_frame_new(__label);\
+ gtk_container_set_border_width(GTK_CONTAINER(frame), 5);\
+ gtk_box_pack_start(GTK_BOX(page), frame, FALSE, FALSE, 0);
+ #define TABLE_NEW(__rows, __cols) table = gtk_table_new(__rows, __cols, FALSE);\
+ gtk_container_set_border_width(GTK_CONTAINER(table), 5);\
+ gtk_container_add(GTK_CONTAINER(frame), table);
+ #define WIDGET_ADD(__widget, __left, __right, __top, __bottom)\
+ gtk_table_attach(GTK_TABLE(table), __widget\
+ , __left, __right, __top, __bottom\
+ , 0, GTK_FILL, 8, 2)
+ #define FILLED_ADD(__widget, __left, __right, __top, __bottom)\
+ gtk_table_attach(GTK_TABLE(table), __widget\
+ , __left, __right, __top, __bottom\
+ , GTK_EXPAND | GTK_FILL, GTK_FILL, 8, 2)
+ #define INDENTED_ADD(__widget, __left, __right, __top, __bottom)\
+ align = gtk_alignment_new(0, 0.5, 0, 0);\
+ gtk_container_add(GTK_CONTAINER(align), __widget);\
+ gtk_size_group_add_widget(sizegroup, align);\
+ WIDGET_ADD(align, __left, __right, __top, __bottom)
+ #define SEMI_INDENTED_ADD(__widget, __left, __right, __top, __bottom)\
+ align = gtk_alignment_new(0, 0.5, 0, 0);\
+ gtk_container_add(GTK_CONTAINER(align), __widget);\
+ gtk_size_group_add_widget(sizegroup2, align);\
+ WIDGET_ADD(align, __left, __right, __top, __bottom)
+ #define SPANNED_ADD(__widget, __left, __right, __top, __bottom)\
+ align = gtk_alignment_new(0, 0.5, 0, 0);\
+ gtk_container_add(GTK_CONTAINER(align), __widget);\
+ FILLED_ADD(align, __left, __right, __top, __bottom)
+ // Page "General"
+ PAGE_NEW("General");
+ FRAME_NEW("Startup");
+ TABLE_NEW(2, 2);
+ INDENTED_ADD(gtk_label_new("Load on startup"), 0, 1, 0, 1);
+ combobox = gtk_combo_box_new_text();
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , "Blank page", "Homepage", "Last open pages", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), config->startup);
+ g_signal_connect(combobox, "changed"
+ , G_CALLBACK(on_prefs_loadonstartup_changed), prefs);
+ FILLED_ADD(combobox, 1, 2, 0, 1);
+ INDENTED_ADD(gtk_label_new("Homepage"), 0, 1, 1, 2);
+ entry = gtk_entry_new();
+ gtk_entry_set_text(GTK_ENTRY(entry), config->homepage);
+ g_signal_connect(entry, "focus-out-event"
+ , G_CALLBACK(on_prefs_homepage_focus_out), prefs);
+ FILLED_ADD(entry, 1, 2, 1, 2);
+ // TODO: We need something like "use current website"
+ FRAME_NEW("Downloads");
+ TABLE_NEW(1, 2);
+ INDENTED_ADD(gtk_label_new("Download folder"), 0, 1, 0, 1);
+ GtkWidget* filebutton = gtk_file_chooser_button_new(
+ "Choose download folder", GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER);
+ // FIXME: The default should probably be ~/Desktop
+ gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(filebutton)
+ , g_get_home_dir()); //...
+ gtk_widget_set_sensitive(filebutton, FALSE); //...
+ FILLED_ADD(filebutton, 1, 2, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic
+ ("Show a notification window for finished downloads");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 2, 1, 2);
+ FRAME_NEW("Languages");
+ TABLE_NEW(1, 2);
+ INDENTED_ADD(gtk_label_new("Preferred languages"), 0, 1, 0, 1);
+ entry = gtk_entry_new();
+ // TODO: Make sth like get_browser_languages_default filtering encodings and C out
+ // TODO: Provide a real ui with real language names (iso-codes)
+ const gchar* const* sLanguages = g_get_language_names();
+ gchar* sLanguagesPreferred = g_strjoinv(",", (gchar**)sLanguages);
+ gtk_entry_set_text(GTK_ENTRY(entry), sLanguagesPreferred/*config->sLanguagesPreferred*/);
+ g_free(sLanguagesPreferred);
+ gtk_widget_set_sensitive(entry, FALSE); //...
+ FILLED_ADD(entry, 1, 2, 0, 1);
+
+ // Page "Appearance"
+ PAGE_NEW("Appearance");
+ FRAME_NEW("Font settings");
+ TABLE_NEW(5, 2);
+ INDENTED_ADD(gtk_label_new("Standard font"), 0, 1, 0, 1);
+ button = gtk_font_button_new_with_font("Sans 10"/*config->sFontStandard*/);
+ gtk_widget_set_sensitive(button, FALSE); //...
+ FILLED_ADD(button, 1, 2, 0, 1);
+ INDENTED_ADD(gtk_label_new("Minimum font size"), 0, 1, 1, 2);
+ hbox = gtk_hbox_new(FALSE, 4);
+ spinbutton = gtk_spin_button_new_with_range(5, 12, 1);
+ gtk_spin_button_set_value(GTK_SPIN_BUTTON(spinbutton), 5/*config->iFontSizeMin*/);
+ gtk_widget_set_sensitive(spinbutton, FALSE); //...
+ gtk_box_pack_start(GTK_BOX(hbox), spinbutton, FALSE, FALSE, 0);
+ button = gtk_button_new_with_mnemonic("_Advanced");
+ gtk_widget_set_sensitive(button, FALSE); //...
+ gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, FALSE, 4);
+ FILLED_ADD(hbox, 1, 2, 1, 2);
+ INDENTED_ADD(gtk_label_new("Default encoding"), 0, 1, 2, 3);
+ combobox = gtk_combo_box_new_text();
+ const gchar* encoding = NULL; g_get_charset(&encoding);
+ // TODO: Fallback to utf-8 if the encoding is not sane (e.g. when lang=C)
+ gchar* sEncodingDefault = g_strdup_printf("System (%s)", encoding);
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , sEncodingDefault, "Chinese", "Greek", "Japanese (SHIFT_JIS)"
+ , "Korean", "Russian", "Unicode (UTF-8)", "Western (ISO-8859-1)", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), 0); //...
+ gtk_widget_set_sensitive(combobox, FALSE); //...
+ FILLED_ADD(combobox, 1, 2, 2, 3);
+ button = gtk_button_new_with_label("Advanced settings");
+ gtk_widget_set_sensitive(button, FALSE); //...
+ WIDGET_ADD(button, 1, 2, 2, 3);
+ FRAME_NEW("Default colors");
+ TABLE_NEW(2, 4);
+ SEMI_INDENTED_ADD(gtk_label_new("Text color"), 0, 1, 0, 1);
+ colorbutton = gtk_color_button_new();
+ gtk_widget_set_sensitive(colorbutton, FALSE); //...
+ WIDGET_ADD(colorbutton, 1, 2, 0, 1);
+ SEMI_INDENTED_ADD(gtk_label_new("Background color"), 2, 3, 0, 1);
+ colorbutton = gtk_color_button_new();
+ gtk_widget_set_sensitive(colorbutton, FALSE); //...
+ WIDGET_ADD(colorbutton, 3, 4, 0, 1);
+ SEMI_INDENTED_ADD(gtk_label_new("Normal link color"), 0, 1, 1, 2);
+ colorbutton = gtk_color_button_new();
+ gtk_widget_set_sensitive(colorbutton, FALSE); //...
+ WIDGET_ADD(colorbutton, 1, 2, 1, 2);
+ SEMI_INDENTED_ADD(gtk_label_new("Visited link color"), 2, 3, 1, 2);
+ colorbutton = gtk_color_button_new();
+ gtk_widget_set_sensitive(colorbutton, FALSE); //...
+ WIDGET_ADD(colorbutton, 3, 4, 1, 2);
+
+ // Page "Behavior"
+ PAGE_NEW("Behavior");
+ FRAME_NEW("Browsing");
+ TABLE_NEW(3, 2);
+ INDENTED_ADD(gtk_label_new_with_mnemonic("Open _new pages in"), 0, 1, 0, 1);
+ combobox = gtk_combo_box_new_text();
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , "New tab", "New window", "Current tab", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), config->newPages);
+ g_signal_connect(combobox, "changed"
+ , G_CALLBACK(on_prefs_newpages_changed), prefs);
+ gtk_widget_set_sensitive(combobox, FALSE); //...
+ FILLED_ADD(combobox, 1, 2, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic("Open tabs in the _background");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->openTabsInTheBackground);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_openTabsInTheBackground_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 2, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Open _popups in tabs");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->openPopupsInTabs);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_openPopupsInTabs_toggled), prefs);
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 2, 2, 3);
+ FRAME_NEW("Features");
+ TABLE_NEW(3, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Load _images automatically");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->loadImagesAutomatically);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_loadImagesAutomatically_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 1, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic("_Shrink images to fit");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->shrinkImagesToFit);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_shrinkImagesToFit_toggled), prefs);
+ SPANNED_ADD(checkbutton, 1, 2, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic("_Resizable textareas");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->resizableTextAreas);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_resizableTextAreas_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 1, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Enable java_script");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->enableJavaScript);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_enableJavaScript_toggled), prefs);
+ SPANNED_ADD(checkbutton, 1, 2, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Enable _plugins");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->enablePlugins);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_enablePlugins_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 1, 2, 3);
+ // For now we check for "plugins-enabled", in case this build has no properties
+ if(!g_object_class_find_property(G_OBJECT_GET_CLASS(browser->webView), "plugins-enabled"))
+ gtk_widget_set_sensitive(frame, FALSE);
+
+ // Page "Interface"
+ PAGE_NEW("Interface");
+ FRAME_NEW("Navigationbar");
+ TABLE_NEW(3, 2);
+ INDENTED_ADD(gtk_label_new_with_mnemonic("_Toolbar style"), 0, 1, 0, 1);
+ combobox = gtk_combo_box_new_text();
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , "Default", "Icons", "Text", "Both", "Both horizontal", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), config->toolbarStyle);
+ g_signal_connect(combobox, "changed"
+ , G_CALLBACK(on_prefs_toolbarstyle_changed), prefs);
+ FILLED_ADD(combobox, 1, 2, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic("Show small _icons");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->toolbarSmall);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_toolbarSmall_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 1, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Show web_search");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->toolbarWebSearch);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_toolbarWebSearch_toggled), prefs);
+ SPANNED_ADD(checkbutton, 1, 2, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Show _New Tab Button");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->toolbarNewTab);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_toolbarNewTab_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 1, 2, 3);
+ checkbutton = gtk_check_button_new_with_mnemonic("Show _closed tabs button");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->toolbarClosedTabs);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_toolbarClosedTabs_toggled), prefs);
+ SPANNED_ADD(checkbutton, 1, 2, 2, 3);
+ FRAME_NEW("Miscellaneous");
+ TABLE_NEW(3, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic
+ ("Show close _buttons on tabs");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbutton), config->tabClose);
+ g_signal_connect(checkbutton, "toggled"
+ , G_CALLBACK(on_prefs_closeButtonsOnTabs_toggled), prefs);
+ SPANNED_ADD(checkbutton, 0, 2, 0, 1);
+ INDENTED_ADD(gtk_label_new_with_mnemonic("Tabbar _placement"), 0, 1, 1, 2);
+ combobox = gtk_combo_box_new_text();
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , "Left", "Top", "Right", "Bottom", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), 1); //...
+ gtk_widget_set_sensitive(combobox, FALSE); //...
+ FILLED_ADD(combobox, 1, 2, 1, 2);
+ INDENTED_ADD(gtk_label_new_with_mnemonic("_Location search engine"), 0, 1, 2, 3);
+ entry = gtk_entry_new();
+ gtk_entry_set_text(GTK_ENTRY(entry), config->locationSearch);
+ g_signal_connect(entry, "focus-out-event"
+ , G_CALLBACK(on_prefs_locationsearch_focus_out), prefs);
+ FILLED_ADD(entry, 1, 2, 2, 3);
+
+ // Page "Network"
+ PAGE_NEW("Network");
+ FRAME_NEW("Proxy Server");
+ TABLE_NEW(5, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("_Custom proxy server");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 2, 0, 1);
+ hbox = gtk_hbox_new(FALSE, 4);
+ INDENTED_ADD(gtk_label_new_with_mnemonic("_Host/ Port"), 0, 1, 1, 2);
+ entry = gtk_entry_new();
+ gtk_widget_set_sensitive(entry, FALSE); //...
+ gtk_box_pack_start(GTK_BOX(hbox), entry, FALSE, FALSE, 0);
+ spinbutton = gtk_spin_button_new_with_range(0, 65535, 1);
+ gtk_widget_set_sensitive(spinbutton, FALSE); //...
+ gtk_box_pack_start(GTK_BOX(hbox), spinbutton, FALSE, FALSE, 0);
+ FILLED_ADD(hbox, 1, 2, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic
+ ("Proxy requires authentication");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ // TODO: The proxy user and pass need to be indented further
+ SPANNED_ADD(checkbutton, 0, 2, 2, 3);
+ INDENTED_ADD(gtk_label_new("Username"), 0, 1, 3, 4);
+ entry = gtk_entry_new();
+ gtk_widget_set_sensitive(entry, FALSE); //...
+ FILLED_ADD(entry, 1, 2, 3, 4);
+ INDENTED_ADD(gtk_label_new("Password"), 0, 1, 4, 5);
+ entry = gtk_entry_new();
+ gtk_widget_set_sensitive(entry, FALSE); //...
+ FILLED_ADD(entry, 1, 2, 4, 5);
+ FRAME_NEW("Cache");
+ TABLE_NEW(1, 2);
+ INDENTED_ADD(gtk_label_new("Cache size"), 0, 1, 0, 1);
+ hbox = gtk_hbox_new(FALSE, 4);
+ spinbutton = gtk_spin_button_new_with_range(0, 10000, 10);
+ gtk_spin_button_set_value(GTK_SPIN_BUTTON(spinbutton), 100/*config->iCacheSize*/);
+ gtk_widget_set_sensitive(spinbutton, FALSE); //...
+ gtk_box_pack_start(GTK_BOX(hbox), spinbutton, FALSE, FALSE, 0);
+ gtk_box_pack_start(GTK_BOX(hbox), gtk_label_new("MB"), FALSE, FALSE, 0);
+ button = gtk_button_new_with_label("Clear cache");
+ gtk_widget_set_sensitive(button, FALSE); //...
+ gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, FALSE, 4);
+ FILLED_ADD(hbox, 1, 2, 0, 1);
+
+ // Page "Privacy"
+ PAGE_NEW("Privacy");
+ FRAME_NEW("Cookies");
+ TABLE_NEW(3, 2);
+ INDENTED_ADD(gtk_label_new("Accept cookies"), 0, 1, 0, 1);
+ combobox = gtk_combo_box_new_text();
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , "All cookies", "Session cookies", "None", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), 0); //...
+ gtk_widget_set_sensitive(combobox, FALSE); //...
+ FILLED_ADD(combobox, 1, 2, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic
+ ("Allow cookies from the original website only");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 2, 1, 2);
+ INDENTED_ADD(gtk_label_new("Maximum cookie age"), 0, 1, 2, 3);
+ hbox = gtk_hbox_new(FALSE, 4);
+ spinbutton = gtk_spin_button_new_with_range(0, 360, 1);
+ gtk_spin_button_set_value(GTK_SPIN_BUTTON(spinbutton), 30/*config->iCookieAgeMax*/);
+ gtk_widget_set_sensitive(spinbutton, FALSE); //...
+ gtk_box_pack_start(GTK_BOX(hbox), spinbutton, FALSE, FALSE, 0);
+ gtk_box_pack_start(GTK_BOX(hbox), gtk_label_new("days"), FALSE, FALSE, 0);
+ button = gtk_button_new_with_label("View cookies");
+ gtk_widget_set_sensitive(button, FALSE); //...
+ gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, FALSE, 4);
+ FILLED_ADD(hbox, 1, 2, 2, 3);
+ FRAME_NEW("History");
+ TABLE_NEW(3, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic("Remember my visited pages");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 1, 0, 1);
+ hbox = gtk_hbox_new(FALSE, 4);
+ spinbutton = gtk_spin_button_new_with_range(0, 360, 1);
+ gtk_spin_button_set_value(GTK_SPIN_BUTTON(spinbutton), 30/*config->iHistoryAgeMax*/);
+ gtk_widget_set_sensitive(spinbutton, FALSE); //...
+ gtk_box_pack_start(GTK_BOX(hbox), spinbutton, FALSE, FALSE, 0);
+ gtk_box_pack_start(GTK_BOX(hbox), gtk_label_new("days"), FALSE, FALSE, 0);
+ SPANNED_ADD(hbox, 1, 2, 0, 1);
+ checkbutton = gtk_check_button_new_with_mnemonic
+ ("Remember my form inputs");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 2, 1, 2);
+ checkbutton = gtk_check_button_new_with_mnemonic
+ ("Remember my downloaded files");
+ gtk_widget_set_sensitive(checkbutton, FALSE); //...
+ SPANNED_ADD(checkbutton, 0, 2, 2, 3);
+
+ // Page "Programs"
+ PAGE_NEW("Programs");
+ FRAME_NEW("External programs");
+ TABLE_NEW(3, 2);
+ GtkWidget* treeview; GtkTreeViewColumn* column;
+ GtkCellRenderer* renderer_text; GtkCellRenderer* renderer_pixbuf;
+ GtkListStore* liststore = gtk_list_store_new(PROTOCOLS_COL_N
+ , G_TYPE_STRING, G_TYPE_STRING);
+ treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(liststore));
+ prefs->treeview = treeview;
+ renderer_text = gtk_cell_renderer_text_new();
+ renderer_pixbuf = gtk_cell_renderer_pixbuf_new();
+ column = gtk_tree_view_column_new_with_attributes(
+ "Protocol", renderer_text, "text", PROTOCOLS_COL_NAME, NULL);
+ gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
+ column = gtk_tree_view_column_new();
+ gtk_tree_view_column_set_title(column, "Command");
+ gtk_tree_view_column_pack_start(column, renderer_pixbuf, FALSE);
+ gtk_tree_view_column_set_cell_data_func(column, renderer_pixbuf
+ , (GtkTreeCellDataFunc)on_prefs_protocols_render_icon, prefs, NULL);
+ renderer_text = gtk_cell_renderer_text_new();
+ g_object_set(G_OBJECT(renderer_text), "editable", TRUE, NULL);
+ g_signal_connect(GTK_OBJECT(renderer_text), "edited"
+ , G_CALLBACK(on_prefs_protocols_edited), prefs);
+ gtk_tree_view_column_pack_start(column, renderer_text, TRUE);
+ gtk_tree_view_column_add_attribute(column, renderer_text, "text", PROTOCOLS_COL_COMMAND);
+ gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
+ GtkWidget* scrolled = gtk_scrolled_window_new(NULL, NULL);
+ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled)
+ , GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
+ gtk_container_add(GTK_CONTAINER(scrolled), treeview);
+ gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
+ hbox = gtk_hbox_new(FALSE, 0);
+ gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 5);
+ guint i;
+ for(i = 0; i < config->protocols_names->len; i++)
+ {
+ gchar* protocol = (gchar*)g_ptr_array_index(config->protocols_names, i);
+ // TODO: We might want to determine 'default' programs somehow
+ // TODO: Any way to make it easier to add eg. only a mail client? O_o
+ const gchar* command = g_datalist_get_data(&config->protocols_commands, protocol);
+ gtk_list_store_insert_with_values(GTK_LIST_STORE(liststore), NULL, i
+ , PROTOCOLS_COL_NAME , protocol
+ , PROTOCOLS_COL_COMMAND, command
+ , -1);
+ }
+ g_object_unref(liststore);
+ GtkWidget* vbox = gtk_vbox_new(FALSE, 4);
+ gtk_box_pack_start(GTK_BOX(hbox), vbox, FALSE, FALSE, 4);
+ combobox = gtk_combo_box_new_text();
+ prefs->combobox = combobox;
+ sokoke_combo_box_add_strings(GTK_COMBO_BOX(combobox)
+ , "download", "ed2k", "feed", "ftp", "irc", "mailto"
+ , "news", "tel", "torrent", "view-source", NULL);
+ gtk_combo_box_set_active(GTK_COMBO_BOX(combobox), 1);
+ gtk_box_pack_start(GTK_BOX(vbox), combobox, FALSE, FALSE, 0);
+ button = gtk_button_new_from_stock(GTK_STOCK_ADD);
+ prefs->add = button;
+ g_signal_connect(combobox, "changed"
+ , G_CALLBACK(on_prefs_protocols_combobox_changed), prefs);
+ g_signal_connect(button, "clicked"
+ , G_CALLBACK(on_prefs_protocols_add_clicked), prefs);
+ gtk_box_pack_start(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ button = gtk_label_new(""); // This is an invisible separator
+ gtk_box_pack_start(GTK_BOX(vbox), button, TRUE, TRUE, 12);
+ button = gtk_button_new_from_stock(GTK_STOCK_REMOVE);
+ gtk_widget_set_sensitive(button, FALSE); //...
+ gtk_box_pack_end(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ FILLED_ADD(hbox, 0, 2, 0, 2);
+ gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox)
+ , notebook, FALSE, FALSE, 4);
+ gtk_widget_show_all(GTK_DIALOG(dialog)->vbox);
+ return dialog;
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __PREFS_H__
+#define __PREFS_H__ 1
+
+#include "browser.h"
+
+#include <gtk/gtk.h>
+
+// -- Types
+
+typedef struct
+{
+ CBrowser* browser;
+ //GtkWidget* window;
+ GtkWidget* treeview;
+ GtkWidget* combobox;
+ GtkWidget* add;
+} CPrefs;
+
+enum
+{
+ PROTOCOLS_COL_NAME,
+ PROTOCOLS_COL_COMMAND,
+ PROTOCOLS_COL_N
+};
+
+// -- Declarations
+
+GtkWidget*
+prefs_preferences_dialog_new(CBrowser*);
+
+#endif /* !__PREFS_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "search.h"
+
+#include "sokoke.h"
+
+#include <stdio.h>
+#include <string.h>
+
+GList* search_engines_new(void)
+{
+ return NULL;
+}
+
+void search_engines_free(GList* searchEngines)
+{
+ g_list_foreach(searchEngines, (GFunc)search_engine_free, NULL);
+ g_list_free(searchEngines);
+}
+
+gboolean search_engines_from_file(GList** searchEngines, const gchar* filename
+ , GError** error)
+{
+ g_return_val_if_fail(!g_list_nth(*searchEngines, 0), FALSE);
+ GKeyFile* keyFile = g_key_file_new();
+ g_key_file_load_from_file(keyFile, filename, G_KEY_FILE_KEEP_COMMENTS, error);
+ /*g_key_file_load_from_data_dirs(keyFile, sFilename, NULL
+ , G_KEY_FILE_KEEP_COMMENTS, error);*/
+ gchar** engines = g_key_file_get_groups(keyFile, NULL);
+ guint i;
+ for(i = 0; engines[i] != NULL; i++)
+ {
+ SearchEngine* engine = search_engine_new();
+ search_engine_set_short_name(engine, engines[i]);
+ engine->description = g_key_file_get_string(keyFile, engines[i], "description", NULL);
+ engine->url = g_key_file_get_string(keyFile, engines[i], "url", NULL);
+ engine->inputEncoding = g_key_file_get_string(keyFile, engines[i], "input-encoding", NULL);
+ engine->icon = g_key_file_get_string(keyFile, engines[i], "icon", NULL);
+ engine->keyword = g_key_file_get_string(keyFile, engines[i], "keyword", NULL);
+ *searchEngines = g_list_prepend(*searchEngines, engine);
+ }
+ *searchEngines = g_list_reverse(*searchEngines);
+ g_strfreev(engines);
+ g_key_file_free(keyFile);
+ return !(error && *error);
+}
+
+static void key_file_set_string(GKeyFile* keyFile, const gchar* group
+ , const gchar* key, const gchar* string)
+{
+ g_return_if_fail(group);
+ if(string)
+ g_key_file_set_string(keyFile, group, key, string);
+}
+
+gboolean search_engines_to_file(GList* searchEngines, const gchar* filename
+ , GError** error)
+{
+ GKeyFile* keyFile = g_key_file_new();
+ guint n = g_list_length(searchEngines);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ SearchEngine* engine = (SearchEngine*)g_list_nth_data(searchEngines, i);
+ const gchar* name = search_engine_get_short_name(engine);
+ key_file_set_string(keyFile, name, "description", engine->description);
+ key_file_set_string(keyFile, name, "url", engine->url);
+ key_file_set_string(keyFile, name, "input-encoding", engine->inputEncoding);
+ key_file_set_string(keyFile, name, "icon", engine->icon);
+ key_file_set_string(keyFile, name, "keyword", engine->keyword);
+ }
+ gboolean bSaved = sokoke_key_file_save_to_file(keyFile, filename, error);
+ g_key_file_free(keyFile);
+
+ return bSaved;
+}
+
+SearchEngine* search_engine_new()
+{
+ SearchEngine* engine = g_new(SearchEngine, 1);
+ engine->shortName = g_strdup("");
+ engine->description = NULL;
+ engine->url = NULL;
+ engine->inputEncoding = NULL;
+ engine->icon = NULL;
+ engine->keyword = NULL;
+ return engine;
+}
+
+void search_engine_free(SearchEngine* engine)
+{
+ g_return_if_fail(engine);
+ g_free(engine->shortName);
+ g_free(engine->description);
+ g_free(engine->url);
+ g_free(engine->inputEncoding);
+ g_free(engine->icon);
+ g_free(engine->keyword);
+ g_free(engine);
+}
+
+SearchEngine* search_engine_copy(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ SearchEngine* copy = search_engine_new();
+ search_engine_set_short_name(copy, engine->shortName);
+ search_engine_set_description(copy, engine->description);
+ search_engine_set_url(copy, engine->url);
+ search_engine_set_input_encoding(copy, engine->inputEncoding);
+ search_engine_set_icon(copy, engine->icon);
+ search_engine_set_keyword(copy, engine->keyword);
+ return engine;
+}
+
+GType search_engine_get_type()
+{
+ static GType type = 0;
+ if(!type)
+ type = g_pointer_type_register_static("search_engine");
+ return type;
+}
+
+G_CONST_RETURN gchar* search_engine_get_short_name(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ return engine->shortName;
+}
+
+G_CONST_RETURN gchar* search_engine_get_description(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ return engine->description;
+}
+
+G_CONST_RETURN gchar* search_engine_get_url(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ return engine->url;
+}
+
+G_CONST_RETURN gchar* search_engine_get_input_encoding(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ return engine->inputEncoding;
+}
+
+G_CONST_RETURN gchar* search_engine_get_icon(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ return engine->icon;
+}
+
+G_CONST_RETURN gchar* search_engine_get_keyword(SearchEngine* engine)
+{
+ g_return_val_if_fail(engine, NULL);
+ return engine->keyword;
+}
+
+void search_engine_set_short_name(SearchEngine* engine, const gchar* shortName)
+{
+ g_return_if_fail(engine);
+ g_return_if_fail(shortName);
+ g_free(engine->shortName);
+ engine->shortName = g_strdup(shortName);
+}
+
+void search_engine_set_description(SearchEngine* engine, const gchar* description)
+{
+ g_return_if_fail(engine);
+ g_free(engine->description);
+ engine->description = g_strdup(description);
+}
+
+void search_engine_set_url(SearchEngine* engine, const gchar* url)
+{
+ g_return_if_fail(engine);
+ g_free(engine->url);
+ engine->url = g_strdup(url);
+}
+
+void search_engine_set_input_encoding(SearchEngine* engine, const gchar* inputEncoding)
+{
+ g_return_if_fail(engine);
+ g_free(engine->inputEncoding);
+ engine->inputEncoding = g_strdup(inputEncoding);
+}
+
+void search_engine_set_icon(SearchEngine* engine, const gchar* icon)
+{
+ g_return_if_fail(engine);
+ g_free(engine->icon);
+ engine->icon = g_strdup(icon);
+}
+
+void search_engine_set_keyword(SearchEngine* engine, const gchar* keyword)
+{
+ g_return_if_fail(engine);
+ g_free(engine->keyword);
+ engine->keyword = g_strdup(keyword);
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __SEARCH_H__
+#define __SEARCH_H__ 1
+
+#include <glib.h>
+#include <glib-object.h>
+
+// Note: This structure is entirely private.
+typedef struct
+{
+ gchar* shortName;
+ gchar* description;
+ gchar* url;
+ gchar* inputEncoding;
+ gchar* icon;
+ gchar* keyword;
+} SearchEngine;
+
+GList*
+search_engines_new(void);
+
+void
+search_engines_free(GList*);
+
+gboolean
+search_engines_from_file(GList**, const gchar*, GError**);
+
+gboolean
+search_engines_to_file(GList*, const gchar*, GError**);
+
+SearchEngine*
+search_engine_new(void);
+
+void
+search_engine_free(SearchEngine*);
+
+SearchEngine*
+search_engine_copy(SearchEngine*);
+
+GType
+search_engine_get_type();
+
+#define G_TYPE_SEARCH_ENGINE search_engine_get_type()
+
+G_CONST_RETURN gchar*
+search_engine_get_short_name(SearchEngine*);
+
+G_CONST_RETURN gchar*
+search_engine_get_description(SearchEngine*);
+
+G_CONST_RETURN gchar*
+search_engine_get_url(SearchEngine*);
+
+G_CONST_RETURN gchar*
+search_engine_get_input_encoding(SearchEngine*);
+
+G_CONST_RETURN gchar*
+search_engine_get_icon(SearchEngine*);
+
+G_CONST_RETURN gchar*
+search_engine_get_keyword(SearchEngine*);
+
+void
+search_engine_set_short_name(SearchEngine*, const gchar*);
+
+void
+search_engine_set_description(SearchEngine*, const gchar*);
+
+void
+search_engine_set_url(SearchEngine*, const gchar*);
+
+void
+search_engine_set_input_encoding(SearchEngine*, const gchar*);
+
+void
+search_engine_set_icon(SearchEngine*, const gchar*);
+
+void
+search_engine_set_keyword(SearchEngine*, const gchar*);
+
+#endif /* !__SEARCH_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "sokoke.h"
+
+#include "debug.h"
+
+#include <string.h>
+#ifdef HAVE_UNISTD_H
+ #include <unistd.h>
+#endif
+#include <gdk/gdkkeysyms.h>
+
+void sokoke_combo_box_add_strings(GtkComboBox* combobox
+ , const gchar* sLabelFirst, ...)
+{
+ // Add a number of strings to a combobox, terminated with NULL
+ // This works only for text comboboxes
+ va_list args;
+ va_start(args, sLabelFirst);
+
+ const gchar* sLabel;
+ for(sLabel = sLabelFirst; sLabel; sLabel = va_arg(args, const gchar*))
+ gtk_combo_box_append_text(combobox, sLabel);
+
+ va_end(args);
+}
+
+void sokoke_radio_action_set_current_value(GtkRadioAction* action
+ , gint current_value)
+{
+ // Activates the group member with the given value
+ #if GTK_CHECK_VERSION(2, 10, 0)
+ gtk_radio_action_set_current_value(action, current_value);
+ #else
+ // TODO: Implement this for older gtk
+ UNIMPLEMENTED
+ #endif
+}
+
+void sokoke_widget_set_visible(GtkWidget* widget, gboolean bVisibility)
+{
+ // Show or hide the widget
+ if(bVisibility)
+ gtk_widget_show(widget);
+ else
+ gtk_widget_hide(widget);
+}
+
+void sokoke_container_show_children(GtkContainer* container)
+{
+ // Show every child but not the container itself
+ gtk_container_foreach(container, (GtkCallback)(gtk_widget_show_all), NULL);
+}
+
+void sokoke_widget_set_tooltip_text(GtkWidget* widget, const gchar* sText)
+{
+ #if GTK_CHECK_VERSION(2, 12, 0)
+ gtk_widget_set_tooltip_text(widget, sText);
+ #else
+ static GtkTooltips* tooltips;
+ if(!tooltips)
+ tooltips = gtk_tooltips_new();
+ gtk_tooltips_set_tip(tooltips, widget, sText, NULL);
+ #endif
+}
+
+void sokoke_tool_item_set_tooltip_text(GtkToolItem* toolitem, const gchar* sText)
+{
+ // TODO: Use 2.12 api if available
+ GtkTooltips* tooltips = gtk_tooltips_new();
+ gtk_tool_item_set_tooltip(toolitem, tooltips, sText, NULL);
+}
+
+void sokoke_widget_popup(GtkWidget* widget, GtkMenu* menu
+ , GdkEventButton* event)
+{
+ // TODO: Provide a GtkMenuPositionFunc in case a keyboard invoked this
+ int button, event_time;
+ if(event)
+ {
+ button = event->button;
+ event_time = event->time;
+ }
+ else
+ {
+ button = 0;
+ event_time = gtk_get_current_event_time();
+ }
+
+ if(!gtk_menu_get_attach_widget(menu))
+ gtk_menu_attach_to_widget(menu, widget, NULL);
+ gtk_menu_popup(menu, NULL, NULL, NULL, NULL, button, event_time);
+}
+
+enum
+{
+ SOKOKE_DESKTOP_UNKNOWN,
+ SOKOKE_DESKTOP_XFCE
+};
+
+static guint sokoke_get_desktop(void)
+{
+ // Are we running in Xfce?
+ gint iResult; gchar* stdout; gchar* stderr;
+ gboolean bSuccess = g_spawn_command_line_sync(
+ "xprop -root _DT_SAVE_MODE | grep -q xfce4"
+ , &stdout, &stderr, &iResult, NULL);
+ if(bSuccess && !iResult)
+ return SOKOKE_DESKTOP_XFCE;
+
+ return SOKOKE_DESKTOP_UNKNOWN;
+}
+
+gpointer sokoke_xfce_header_new(const gchar* sIcon, const gchar* sTitle)
+{
+
+ // Create an xfce header with icon and title
+ // This returns NULL if the desktop is not xfce
+ if(sokoke_get_desktop() == SOKOKE_DESKTOP_XFCE)
+ {
+ GtkWidget* entry = gtk_entry_new();
+ gchar* sMarkup;
+ GtkWidget* xfce_heading = gtk_event_box_new();
+ gtk_widget_modify_bg(xfce_heading, GTK_STATE_NORMAL
+ , &entry->style->base[GTK_STATE_NORMAL]);
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 12);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 6);
+ GtkWidget* icon = gtk_image_new_from_icon_name(sIcon, GTK_ICON_SIZE_DIALOG);
+ gtk_box_pack_start(GTK_BOX(hbox), icon, FALSE, FALSE, 0);
+ GtkWidget* label = gtk_label_new(NULL);
+ gtk_widget_modify_fg(label, GTK_STATE_NORMAL
+ , &entry->style->text[GTK_STATE_NORMAL]);
+ sMarkup = g_strdup_printf("<span size='large' weight='bold'>%s</span>", sTitle);
+ gtk_label_set_markup(GTK_LABEL(label), sMarkup);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ gtk_container_add(GTK_CONTAINER(xfce_heading), hbox);
+ g_free(sMarkup);
+ return xfce_heading;
+ }
+ return NULL;
+}
+
+gpointer sokoke_superuser_warning_new(void)
+{
+ // Create a horizontal bar with a security warning
+ // This returns NULL if the user is no superuser
+ #ifdef HAVE_UNISTD_H
+ if(G_UNLIKELY(!geteuid())) // effective superuser?
+ {
+ GtkWidget* hbox = gtk_event_box_new();
+ gtk_widget_modify_bg(hbox, GTK_STATE_NORMAL
+ , &hbox->style->bg[GTK_STATE_SELECTED]);
+ GtkWidget* label = gtk_label_new("Warning: You are using the superuser account!");
+ gtk_misc_set_padding(GTK_MISC(label), 0, 2);
+ gtk_widget_modify_fg(GTK_WIDGET(label), GTK_STATE_NORMAL
+ , >K_WIDGET(label)->style->fg[GTK_STATE_SELECTED]);
+ gtk_container_add(GTK_CONTAINER(hbox), GTK_WIDGET(label));
+ return hbox;
+ }
+ #endif
+ return NULL;
+}
+
+GtkWidget* sokoke_hig_frame_new(const gchar* sLabel)
+{
+ // Create a frame with no actual frame but a bold label and indentation
+ GtkWidget* frame = gtk_frame_new(NULL);
+ gchar* sLabelBold = g_strdup_printf("<b>%s</b>", sLabel);
+ GtkWidget* label = gtk_label_new(NULL);
+ gtk_label_set_markup(GTK_LABEL(label), sLabelBold);
+ g_free(sLabelBold);
+ gtk_frame_set_label_widget(GTK_FRAME(frame), label);
+ gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_NONE);
+ return frame;
+}
+
+void sokoke_widget_set_pango_font_style(GtkWidget* widget, PangoStyle style)
+{
+ // Conveniently change the pango font style
+ // For some reason we need to reset if we actually want the normal style
+ if(style == PANGO_STYLE_NORMAL)
+ gtk_widget_modify_font(widget, NULL);
+ else
+ {
+ PangoFontDescription* pangofontdesc = pango_font_description_new();
+ pango_font_description_set_style(pangofontdesc, PANGO_STYLE_ITALIC);
+ gtk_widget_modify_font(widget, pangofontdesc);
+ pango_font_description_free(pangofontdesc);
+ }
+}
+
+static gboolean sokoke_on_entry_focus_in_event(GtkEntry* entry, GdkEventFocus *event
+ , gpointer userdata)
+{
+ gboolean bDefaultText = (gboolean)g_object_get_data(G_OBJECT(entry)
+ , "sokoke_bDefaultText");
+ if(bDefaultText)
+ {
+ gtk_entry_set_text(entry, "");
+ g_object_set_data(G_OBJECT(entry), "sokoke_bDefaultText", (gpointer)FALSE);
+ sokoke_widget_set_pango_font_style(GTK_WIDGET(entry), PANGO_STYLE_NORMAL);
+ }
+ return FALSE;
+}
+
+static gboolean sokoke_on_entry_focus_out_event(GtkEntry* entry, GdkEventFocus* event
+ , gpointer userdata)
+{
+ const gchar* sText = gtk_entry_get_text(entry);
+ if(sText[0] == '\0')
+ {
+ const gchar* sDefaultText = (const gchar*)g_object_get_data(
+ G_OBJECT(entry), "sokoke_sDefaultText");
+ gtk_entry_set_text(entry, sDefaultText);
+ g_object_set_data(G_OBJECT(entry), "sokoke_bDefaultText", (gpointer)TRUE);
+ sokoke_widget_set_pango_font_style(GTK_WIDGET(entry), PANGO_STYLE_ITALIC);
+ }
+ return FALSE;
+}
+
+void sokoke_entry_set_default_text(GtkEntry* entry, const gchar* sDefaultText)
+{
+ // Note: The default text initially overwrites any previous text
+ gchar* sOldValue = g_object_get_data(G_OBJECT(entry), "sokoke_sDefaultText");
+ if(!sOldValue)
+ {
+ g_object_set_data(G_OBJECT(entry), "sokoke_bDefaultText", (gpointer)TRUE);
+ sokoke_widget_set_pango_font_style(GTK_WIDGET(entry), PANGO_STYLE_ITALIC);
+ gtk_entry_set_text(entry, sDefaultText);
+ }
+ g_object_set_data(G_OBJECT(entry), "sokoke_sDefaultText", (gpointer)sDefaultText);
+ g_signal_connect(entry, "focus-in-event"
+ , G_CALLBACK(sokoke_on_entry_focus_in_event), NULL);
+ g_signal_connect(entry, "focus-out-event"
+ , G_CALLBACK(sokoke_on_entry_focus_out_event), NULL);
+}
+
+gchar* sokoke_key_file_get_string_default(GKeyFile* key_file
+ , const gchar* group_name, const gchar* key, const gchar* def, GError* *error)
+{
+ gchar* value = g_key_file_get_string(key_file, group_name, key, error);
+ return value == NULL ? g_strdup(def) : value;
+}
+
+gint sokoke_key_file_get_integer_default(GKeyFile* key_file
+ , const gchar* group_name, const gchar* key, const gint def, GError** error)
+{
+ if(!g_key_file_has_key(key_file, group_name, key, NULL))
+ return def;
+ return g_key_file_get_integer(key_file, group_name, key, error);
+}
+
+gboolean sokoke_key_file_save_to_file(GKeyFile* key_file
+ , const gchar* sFilename, GError** error)
+{
+ gchar* sData = g_key_file_to_data(key_file, NULL, error);
+ if(!sData)
+ return FALSE;
+ FILE* fp;
+ if(!(fp = fopen(sFilename, "w")))
+ {
+ *error = g_error_new(G_FILE_ERROR, G_FILE_ERROR_ACCES
+ , "Writing failed.");
+ return FALSE;
+ }
+ fputs(sData, fp);
+ fclose(fp);
+ g_free(sData);
+ return TRUE;
+}
+
+void sokoke_widget_get_text_size(GtkWidget* widget, const gchar* sText
+ , gint* w, gint* h)
+{
+ PangoLayout* layout = gtk_widget_create_pango_layout(widget, sText);
+ pango_layout_get_pixel_size(layout, w, h);
+ g_object_unref(layout);
+}
+
+void sokoke_menu_item_set_accel(GtkMenuItem* menuitem, const gchar* sPath
+ , const gchar* sKey, GdkModifierType accel_mods)
+{
+ if(sPath && *sPath)
+ {
+ gchar* path = g_strconcat("<", g_get_prgname(), ">/", sPath, NULL);
+ gtk_menu_item_set_accel_path(GTK_MENU_ITEM(menuitem), path);
+ guint keyVal = sKey ? gdk_keyval_from_name(sKey) : 0;
+ gtk_accel_map_add_entry(path, keyVal, accel_mods);
+ g_free(path);
+ }
+}
+
+gboolean sokoke_entry_can_undo(GtkEntry* entry)
+{
+ // TODO: Can we undo the last input?
+ return FALSE;
+}
+
+gboolean sokoke_entry_can_redo(GtkEntry* entry)
+{
+ // TODO: Can we redo the last input?
+ return FALSE;
+}
+
+void sokoke_entry_undo(GtkEntry* entry)
+{
+ // TODO: Implement undo
+ UNIMPLEMENTED
+}
+
+void sokoke_entry_redo(GtkEntry* entry)
+{
+ // TODO: Implement redo
+ UNIMPLEMENTED
+}
+
+static gboolean sokoke_on_undo_entry_key_down(GtkEntry* widget, GdkEventKey* event
+ , gpointer userdata)
+{
+ switch(event->keyval)
+ {
+ case GDK_Undo:
+ sokoke_entry_undo(widget);
+ return FALSE;
+ case GDK_Redo:
+ sokoke_entry_redo(widget);
+ return FALSE;
+ default:
+ return FALSE;
+ }
+}
+
+static void sokoke_on_undo_entry_populate_popup(GtkEntry* entry, GtkMenu* menu
+ , gpointer userdata)
+{
+ // Enhance the entry's menu with undo and redo items.
+ GtkWidget* menuitem = gtk_separator_menu_item_new();
+ gtk_menu_shell_prepend((GtkMenuShell*)menu, menuitem);
+ gtk_widget_show(menuitem);
+ menuitem = gtk_image_menu_item_new_from_stock(GTK_STOCK_REDO, NULL);
+ g_signal_connect(menuitem, "activate", G_CALLBACK(sokoke_entry_redo), userdata);
+ gtk_widget_set_sensitive(menuitem, sokoke_entry_can_redo(entry));
+ gtk_menu_shell_prepend((GtkMenuShell*)menu, menuitem);
+ gtk_widget_show(menuitem);
+ menuitem = gtk_image_menu_item_new_from_stock(GTK_STOCK_UNDO, NULL);
+ g_signal_connect(menuitem, "activate", G_CALLBACK(sokoke_entry_undo), userdata);
+ gtk_widget_set_sensitive(menuitem, sokoke_entry_can_undo(entry));
+ gtk_menu_shell_prepend((GtkMenuShell*)menu, menuitem);
+ gtk_widget_show(menuitem);
+}
+
+gboolean sokoke_entry_get_can_undo(GtkEntry* entry)
+{
+ // TODO: Is this entry undo enabled?
+ return FALSE;
+}
+
+void sokoke_entry_set_can_undo(GtkEntry* entry, gboolean bCanUndo)
+{
+ if(bCanUndo)
+ {
+ g_signal_connect(entry, "key-press-event"
+ , G_CALLBACK(sokoke_on_undo_entry_key_down), NULL);
+ g_signal_connect(entry, "populate-popup"
+ , G_CALLBACK(sokoke_on_undo_entry_populate_popup), NULL);
+ }
+ else
+ {
+ ; // TODO: disconnect signal
+ UNIMPLEMENTED
+ }
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __SOKOKE_H__
+#define __SOKOKE_H__ 1
+
+#include <gtk/gtk.h>
+
+// Many themes need this hack for small toolbars to work
+#define GTK_ICON_SIZE_SMALL_TOOLBAR GTK_ICON_SIZE_BUTTON
+
+void
+sokoke_combo_box_add_strings(GtkComboBox* combobox, const gchar* sLabelFirst, ...);
+
+void
+sokoke_radio_action_set_current_value(GtkRadioAction* action, gint current_value);
+
+void
+sokoke_widget_set_visible(GtkWidget* widget, gboolean bVisibility);
+
+void
+sokoke_container_show_children(GtkContainer* container);
+
+void
+sokoke_widget_set_tooltip_text(GtkWidget* widget, const gchar* sText);
+
+void
+sokoke_tool_item_set_tooltip_text(GtkToolItem* toolitem, const gchar* sText);
+
+void
+sokoke_widget_popup(GtkWidget* widget, GtkMenu* menu, GdkEventButton* event);
+
+gpointer
+sokoke_xfce_header_new(const gchar* sIcon, const gchar* sTitle);
+
+gpointer
+sokoke_superuser_warning_new(void);
+
+GtkWidget*
+sokoke_hig_frame_new(const gchar* sLabel);
+
+void
+sokoke_widget_set_pango_font_style(GtkWidget* widget, PangoStyle style);
+
+void
+sokoke_entry_set_default_text(GtkEntry* entry, const gchar* sDefaultText);
+
+gchar*
+sokoke_key_file_get_string_default(GKeyFile* key_file
+ , const gchar* group_name, const gchar* key, const gchar* def, GError* *error);
+
+gint
+sokoke_key_file_get_integer_default(GKeyFile* key_file
+ , const gchar* group_name, const gchar* key, const gint def, GError* *error);
+
+gboolean
+sokoke_key_file_save_to_file(GKeyFile* key_file
+ , const gchar* file, GError* *error);
+
+void
+sokoke_widget_get_text_size(GtkWidget* widget, const gchar* sText
+ , gint* w, gint* h);
+
+void
+sokoke_menu_item_set_accel(GtkMenuItem* menuitem, const gchar* sPath
+ , const gchar* sKey, GdkModifierType accel_mods);
+
+gboolean
+sokoke_entry_can_undo(GtkEntry* entry);
+
+gboolean
+sokoke_entry_can_redo(GtkEntry* entry);
+
+void
+sokoke_entry_undo(GtkEntry* entry);
+
+void
+sokoke_entry_redo(GtkEntry* entry);
+
+gboolean
+sokoke_entry_get_can_undo(GtkEntry* entry);
+
+void
+sokoke_entry_set_can_undo(GtkEntry* entry, gboolean bCanUndo);
+
+#endif /* !__SOKOKE_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __UI_H__
+#define __UI_H__ 1
+
+// -- Credits
+
+static const gchar* credits_authors[] = { "Christian Dywan <christian@twotoasts.de>", NULL };
+static const gchar* credits_documenters/*[]*/ = /*{ */NULL/* }*/;
+static const gchar* credits_artists[] = { "Nancy Runge <nancy@twotoasts.de>", NULL };
+
+// -- Licenses
+
+static const gchar* license =
+ "This library is free software; you can redistribute it and/or\n"
+ "modify it under the terms of the GNU Lesser General Public\n"
+ "License as published by the Free Software Foundation; either\n"
+ "version 2.1 of the License, or (at your option) any later version.\n";
+
+// -- User interface description
+
+static const gchar* ui_markup =
+ "<ui>"
+ "<menubar>"
+ "<menu action='File'>"
+ "<menuitem action='WindowNew'/>"
+ "<menuitem action='TabNew'/>"
+ "<separator/>"
+ "<menuitem action='Open'/>"
+ "<separator/>"
+ "<menuitem action='SaveAs'/>"
+ "<separator/>"
+ "<menuitem action='TabClose'/>"
+ "<menuitem action='WindowClose'/>"
+ "<separator/>"
+ "<menuitem action='PageSetup'/>"
+ "<menuitem action='PrintPreview'/>"
+ "<menuitem action='Print'/>"
+ "<separator/>"
+ "<menuitem action='PrivateBrowsing'/>"
+ "<menuitem action='WorkOffline'/>"
+ "<separator/>"
+ "<menuitem action='Quit'/>"
+ "</menu>"
+ "<menu action='Edit'>"
+ "<menuitem action='Undo'/>"
+ "<menuitem action='Redo'/>"
+ "<separator/>"
+ "<menuitem action='Cut'/>"
+ "<menuitem action='Copy'/>"
+ "<menuitem action='Paste'/>"
+ "<menuitem action='Delete'/>"
+ "<separator/>"
+ "<menuitem action='SelectAll'/>"
+ "<separator/>"
+ "<menuitem action='ManageSearchEngines'/>"
+ "<menuitem action='Preferences'/>"
+ "</menu>"
+ "<menu action='View'>"
+ "<menu action='Toolbars'>"
+ "<menuitem action='ToolbarNavigation'/>"
+ "<menuitem action='ToolbarBookmarks'/>"
+ "<menuitem action='ToolbarDownloads'/>"
+ "<menuitem action='ToolbarStatus'/>"
+ "</menu>"
+ "<menuitem action='Panels'/>"
+ "<separator/>"
+ "<menuitem action='Refresh'/>"
+ "<menuitem action='Stop'/>"
+ "<menu action='RefreshEvery'>"
+ "<menuitem action='RefreshEveryEnable'/>"
+ "<separator/>"
+ "<menuitem action='RefreshEvery30'/>"
+ "<menuitem action='RefreshEvery60'/>"
+ "<menuitem action='RefreshEvery300'/>"
+ "<menuitem action='RefreshEvery900'/>"
+ "<menuitem action='RefreshEvery1800'/>"
+ "<menuitem action='RefreshEveryCustom'/>"
+ "</menu>"
+ "<separator/>"
+ "<menuitem action='ZoomIn'/>"
+ "<menuitem action='ZoomOut'/>"
+ "<menuitem action='ZoomNormal'/>"
+ "<separator/>"
+ "<menuitem action='BackgroundImage'/>"
+ "<menuitem action='SourceView'/>"
+ "<menuitem action='Properties'/>"
+ "</menu>"
+ "<menu action='Go'>"
+ "<menuitem action='Back'/>"
+ "<menuitem action='Forward'/>"
+ "<menuitem action='Home'/>"
+ "<menuitem action='Location'/>"
+ "<menuitem action='Websearch'/>"
+ "<menuitem action='OpenInPageholder'/>"
+ "<menu action='TabsClosed'>"
+ // Closed tabs shall be prepended here
+ "<separator/>"
+ "<menuitem action='TabsClosedClear'/>"
+ "</menu>"
+ "<separator/>"
+ "<menuitem action='Find'/>"
+ "<menuitem action='FindNext'/>"
+ "<menuitem action='FindPrevious'/>"
+ "<separator/>"
+ "<menuitem action='FormFill'/>"
+ "</menu>"
+ "<menu action='Bookmarks'>"
+ "<menuitem action='BookmarkNew'/>"
+ "<menuitem action='BookmarksManage'/>"
+ "<separator/>"
+ // Bookmarks shall be appended here
+ "</menu>"
+ "<menu action='Tools'>"
+ "<menuitem action='PanelDownloads'/>"
+ "<menuitem action='PanelBookmarks'/>"
+ "<menuitem action='PanelHistory'/>"
+ "<menuitem action='PanelTabs'/>"
+ "<menuitem action='PanelPageholder'/>"
+ "<menuitem action='PanelExtensions'/>"
+ "<menuitem action='PanelConsole'/>"
+ "<separator/>"
+ // TODO: Insert widgets and custom tools here
+ "</menu>"
+ "<menu action='Window'>"
+ "<menuitem action='SessionLoad'/>"
+ "<menuitem action='SessionSave'/>"
+ "<separator/>"
+ "<menuitem action='TabPrevious'/>"
+ "<menuitem action='TabNext'/>"
+ "<menuitem action='TabOverview'/>"
+ "<separator/>"
+ // All open tabs shall be appended here
+ "</menu>"
+ "<menu action='Help'>"
+ "<menuitem action='HelpContents'/>"
+ "<menuitem action='About'/>"
+ "</menu>"
+ "</menubar>"
+ "<toolbar name='toolbar_navigation'>"
+ "<toolitem action='TabNew'/>"
+ "<toolitem action='Back'/>"
+ "<toolitem action='Forward'/>"
+ "<toolitem action='RefreshStop'/>"
+ "<toolitem action='Home'/>"
+ "<toolitem action='FormFill'/>"
+ "<placeholder name='Location'/>"
+ "<placeholder name='WebSearch'/>"
+ "<placeholder name='TabTrash'/>"
+ "</toolbar>"
+ "<toolbar name='toolbar_panels'>"
+ "<toolitem action='PanelDownloads'/>"
+ "<toolitem action='PanelBookmarks'/>"
+ "<toolitem action='PanelHistory'/>"
+ "<toolitem action='PanelTabs'/>"
+ "<toolitem action='PanelPageholder'/>"
+ "<toolitem action='PanelExtensions'/>"
+ "<toolitem action='PanelConsole'/>"
+ "</toolbar>"
+ "<popup name='popup_webView'>"
+ "<menuitem action='Back'/>"
+ "<menuitem action='Forward'/>"
+ "<menuitem action='Refresh'/>"
+ "<menuitem action='Stop'/>"
+ "<menu action='RefreshEvery'>"
+ "<menuitem action='RefreshEveryEnable'/>"
+ "<separator/>"
+ "<menuitem action='RefreshEvery30'/>"
+ "<menuitem action='RefreshEvery60'/>"
+ "<menuitem action='RefreshEvery300'/>"
+ "<menuitem action='RefreshEvery900'/>"
+ "<menuitem action='RefreshEvery1800'/>"
+ "<menuitem action='RefreshEveryCustom'/>"
+ "</menu>"
+ "<separator/>"
+ "<menuitem action='SelectionLinksNewTabs'/>"
+ "<menuitem action='SelectionTextTabNew'/>"
+ "<menuitem action='SelectionTextTabCurrent'/>"
+ "<menuitem action='SelectionTextWindowNew'/>"
+ "<separator/>"
+ "<menuitem action='UndoTabClose'/>"
+ "<menuitem action='SaveAs'/>"
+ "<menuitem action='BookmarkNew'/>"
+ "<menuitem action='Print'/>"
+ "<separator/>"
+ "<menuitem action='SelectAll'/>"
+ "<separator/>"
+ "<menuitem action='BackgroundImage'/>"
+ "<menuitem action='SourceView'/>"
+ "<menuitem action='Properties'/>"
+ "</popup>"
+ "<popup name='popup_element'>"
+ "<menuitem action='LinkTabNew'/>"
+ "<menuitem action='LinkTabCurrent'/>"
+ "<menuitem action='LinkWindowNew'/>"
+ "<separator/>"
+ "<menuitem action='LinkSaveAs'/>"
+ "<menuitem action='LinkSaveWith'/>"
+ "<menuitem action='LinkCopy'/>"
+ "<menuitem action='LinkBookmarkNew'/>"
+ "<separator/>"
+ "<menuitem action='SelectionLinksNewTabs'/>"
+ "<menuitem action='SelectionTextTabNew'/>"
+ "<menuitem action='SelectionTextTabCurrent'/>"
+ "<menuitem action='SelectionTextWindowNew'/>"
+ "<separator/>"
+ "<menuitem action='ImageViewTabNew'/>"
+ "<menuitem action='ImageViewTabCurrent'/>"
+ "<menuitem action='ImageSaveAs'/>"
+ "<menuitem action='ImageSaveWith'/>"
+ "<menuitem action='ImageCopy'/>"
+ "<separator/>"
+ "<menuitem action='ImageViewTabNew'/>"
+ "<menuitem action='ImageViewTabCurrent'/>"
+ "<separator/>"
+ "<menuitem action='Copy_'/>"
+ "<menuitem action='SelectionSearch'/>"
+ "<menuitem action='SelectionSearchWith'/>"
+ "<menuitem action='SelectionSourceView'/>"
+ "<separator/>"
+ "<menuitem action='Properties'/>"
+ "</popup>"
+ "<popup name='popup_editable'>"
+ "<menuitem action='Undo'/>"
+ "<menuitem action='Redo'/>"
+ "<separator/>"
+ "<menuitem action='Cut'/>"
+ "<menuitem action='Copy'/>"
+ "<menuitem action='Paste'/>"
+ "<menuitem action='Delete'/>"
+ "<separator/>"
+ "<menuitem action='SelectAll'/>"
+ "</popup>"
+ "</ui>";
+
+#endif /* !__UI_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "webSearch.h"
+
+#include "global.h"
+#include "helpers.h"
+#include "search.h"
+#include "sokoke.h"
+
+#include <string.h>
+#include <gdk/gdkkeysyms.h>
+
+void update_searchEngine(guint index, CBrowser* browser)
+{
+ guint n = g_list_length(searchEngines);
+ // Display a default icon in case we have no engines
+ if(!n)
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->webSearch), SEXY_ICON_ENTRY_PRIMARY
+ , GTK_IMAGE(gtk_image_new_from_stock(GTK_STOCK_FIND, GTK_ICON_SIZE_MENU)));
+ // Change the icon and default text according to the chosen engine
+ else
+ {
+ // Reset in case the index is out of range
+ if(index >= n)
+ index = 0;
+ SearchEngine* engine = (SearchEngine*)g_list_nth_data(searchEngines, index);
+ GdkPixbuf* pixbuf = load_web_icon(search_engine_get_icon(engine)
+ , GTK_ICON_SIZE_MENU, browser->navibar);
+ sexy_icon_entry_set_icon(SEXY_ICON_ENTRY(browser->webSearch)
+ , SEXY_ICON_ENTRY_PRIMARY, GTK_IMAGE(gtk_image_new_from_pixbuf(pixbuf)));
+ g_object_unref(pixbuf);
+ sokoke_entry_set_default_text(GTK_ENTRY(browser->webSearch)
+ , search_engine_get_short_name(engine));
+ config->searchEngine = index;
+ }
+}
+
+void on_webSearch_engine_activate(GtkWidget* widget, CBrowser* browser)
+{
+ guint index = GPOINTER_TO_UINT(g_object_get_data(G_OBJECT(widget), "engine"));
+ update_searchEngine(index, browser);
+}
+
+void on_webSearch_icon_released(GtkWidget* widget, SexyIconEntryPosition* pos
+ , gint button, CBrowser* browser)
+{
+ GtkWidget* menu = gtk_menu_new();
+ guint n = g_list_length(searchEngines);
+ GtkWidget* menuitem;
+ if(n)
+ {
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ SearchEngine* engine = (SearchEngine*)g_list_nth_data(searchEngines, i);
+ menuitem = gtk_image_menu_item_new_with_label(
+ search_engine_get_short_name(engine));
+ GdkPixbuf* pixbuf = load_web_icon(search_engine_get_icon(engine)
+ , GTK_ICON_SIZE_MENU, menuitem);
+ GtkWidget* icon = gtk_image_new_from_pixbuf(pixbuf);
+ gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem), icon);
+ g_object_unref(pixbuf);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ g_object_set_data(G_OBJECT(menuitem), "engine", GUINT_TO_POINTER(i));
+ g_signal_connect(menuitem, "activate"
+ , G_CALLBACK(on_webSearch_engine_activate), browser);
+ gtk_widget_show(menuitem);
+ }
+ }
+ else
+ {
+ menuitem = gtk_image_menu_item_new_with_label("Empty");
+ gtk_widget_set_sensitive(menuitem, FALSE);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ gtk_widget_show(menuitem);
+ }
+
+ menuitem = gtk_separator_menu_item_new();
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ gtk_widget_show(menuitem);
+ GtkAction* action = gtk_action_group_get_action(
+ browser->actiongroup, "ManageSearchEngines");
+ menuitem = gtk_action_create_menu_item(action);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+ gtk_widget_show(menuitem);
+ sokoke_widget_popup(widget, GTK_MENU(menu), NULL);
+}
+
+static void on_webSearch_engines_render_icon(GtkTreeViewColumn* column
+ , GtkCellRenderer* renderer, GtkTreeModel* model, GtkTreeIter* iter
+ , GtkWidget* treeview)
+{
+ SearchEngine* searchEngine;
+ gtk_tree_model_get(model, iter, ENGINES_COL_ENGINE, &searchEngine, -1);
+
+ // TODO: Would it be better to not do this on every redraw?
+ const gchar* icon = search_engine_get_icon(searchEngine);
+ if(icon)
+ {
+ GdkPixbuf* pixbuf = load_web_icon(icon, GTK_ICON_SIZE_DND, treeview);
+ g_object_set(renderer, "pixbuf", pixbuf, NULL);
+ if(pixbuf)
+ g_object_unref(pixbuf);
+ }
+ else
+ g_object_set(renderer, "pixbuf", NULL, NULL);
+}
+
+static void on_webSearch_engines_render_text(GtkTreeViewColumn* column
+ , GtkCellRenderer* renderer, GtkTreeModel* model, GtkTreeIter* iter
+ , GtkWidget* treeview)
+{
+ SearchEngine* searchEngine;
+ gtk_tree_model_get(model, iter, ENGINES_COL_ENGINE, &searchEngine, -1);
+ const gchar* name = search_engine_get_short_name(searchEngine);
+ const gchar* description = search_engine_get_description(searchEngine);
+ gchar* markup = g_strdup_printf("<b>%s</b>\n%s", name, description);
+ g_object_set(renderer, "markup", markup, NULL);
+ g_free(markup);
+}
+
+static void webSearch_toggle_edit_buttons(gboolean sensitive, CWebSearch* webSearch)
+{
+ gtk_widget_set_sensitive(webSearch->edit, sensitive);
+ gtk_widget_set_sensitive(webSearch->remove, sensitive);
+}
+
+static void on_webSearch_shortName_changed(GtkWidget* widget, GtkWidget* dialog)
+{
+ const gchar* text = gtk_entry_get_text(GTK_ENTRY(widget));
+ gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog)
+ , GTK_RESPONSE_ACCEPT, text && *text);
+}
+
+const gchar* STR_NON_NULL(const gchar* string)
+{
+ return string ? string : "";
+}
+
+static void webSearch_editEngine_dialog_new(gboolean newEngine, CWebSearch* webSearch)
+{
+ GtkWidget* dialog = gtk_dialog_new_with_buttons(
+ newEngine ? "Add search engine" : "Edit search engine"
+ , GTK_WINDOW(webSearch->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR
+ , GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL
+ , newEngine ? GTK_STOCK_ADD : GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog)
+ , newEngine ? GTK_STOCK_ADD : GTK_STOCK_REMOVE);
+ gtk_container_set_border_width(GTK_CONTAINER(dialog), 5);
+ gtk_container_set_border_width(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), 5);
+ GtkSizeGroup* sizegroup = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
+
+ SearchEngine* searchEngine;
+ GtkTreeModel* liststore;
+ GtkTreeIter iter;
+ if(newEngine)
+ {
+ searchEngine = search_engine_new();
+ gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog)
+ , GTK_RESPONSE_ACCEPT, FALSE);
+ }
+ else
+ {
+ GtkTreeSelection* selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(webSearch->treeview));
+ gtk_tree_selection_get_selected(selection, &liststore, &iter);
+ gtk_tree_model_get(liststore, &iter, ENGINES_COL_ENGINE, &searchEngine, -1);
+ }
+
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ GtkWidget* label = gtk_label_new_with_mnemonic("_Name:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_shortName = gtk_entry_new();
+ g_signal_connect(entry_shortName, "changed"
+ , G_CALLBACK(on_webSearch_shortName_changed), dialog);
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_shortName), TRUE);
+ if(!newEngine)
+ gtk_entry_set_text(GTK_ENTRY(entry_shortName)
+ , search_engine_get_short_name(searchEngine));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_shortName, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ label = gtk_label_new_with_mnemonic("_Description:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_description = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_description), TRUE);
+ if(!newEngine)
+ gtk_entry_set_text(GTK_ENTRY(entry_description)
+ , STR_NON_NULL(search_engine_get_description(searchEngine)));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_description, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ label = gtk_label_new_with_mnemonic("_Url:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_url = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_url), TRUE);
+ if(!newEngine)
+ gtk_entry_set_text(GTK_ENTRY(entry_url)
+ , STR_NON_NULL(search_engine_get_url(searchEngine)));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_url, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ label = gtk_label_new_with_mnemonic("_Icon (name or file):");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_icon = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_icon), TRUE);
+ if(!newEngine)
+ gtk_entry_set_text(GTK_ENTRY(entry_icon)
+ , STR_NON_NULL(search_engine_get_icon(searchEngine)));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_icon, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ hbox = gtk_hbox_new(FALSE, 8);
+ gtk_container_set_border_width(GTK_CONTAINER(hbox), 5);
+ label = gtk_label_new_with_mnemonic("_Keyword:");
+ gtk_size_group_add_widget(sizegroup, label);
+ gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ GtkWidget* entry_keyword = gtk_entry_new();
+ gtk_entry_set_activates_default(GTK_ENTRY(entry_keyword), TRUE);
+ if(!newEngine)
+ gtk_entry_set_text(GTK_ENTRY(entry_keyword)
+ , STR_NON_NULL(search_engine_get_keyword(searchEngine)));
+ gtk_box_pack_start(GTK_BOX(hbox), entry_keyword, TRUE, TRUE, 0);
+ gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), hbox);
+ gtk_widget_show_all(hbox);
+
+ gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT);
+ if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
+ {
+ search_engine_set_short_name(searchEngine
+ , gtk_entry_get_text(GTK_ENTRY(entry_shortName)));
+ search_engine_set_description(searchEngine
+ , gtk_entry_get_text(GTK_ENTRY(entry_description)));
+ search_engine_set_url(searchEngine
+ , gtk_entry_get_text(GTK_ENTRY(entry_url)));
+ /*search_engine_set_input_encoding(searchEngine
+ , gtk_entry_get_text(GTK_ENTRY(entry_inputEncoding)));*/
+ search_engine_set_icon(searchEngine
+ , gtk_entry_get_text(GTK_ENTRY(entry_icon)));
+ search_engine_set_keyword(searchEngine
+ , gtk_entry_get_text(GTK_ENTRY(entry_keyword)));
+
+ if(newEngine)
+ {
+ searchEngines = g_list_append(searchEngines, searchEngine);
+ liststore = gtk_tree_view_get_model(GTK_TREE_VIEW(webSearch->treeview));
+ gtk_list_store_append(GTK_LIST_STORE(liststore), &iter);
+ }
+ gtk_list_store_set(GTK_LIST_STORE(liststore), &iter
+ , ENGINES_COL_ENGINE, searchEngine, -1);
+ webSearch_toggle_edit_buttons(TRUE, webSearch);
+ }
+ gtk_widget_destroy(dialog);
+}
+
+static void on_webSearch_add(GtkWidget* widget, CWebSearch* webSearch)
+{
+ webSearch_editEngine_dialog_new(TRUE, webSearch);
+}
+
+static void on_webSearch_edit(GtkWidget* widget, CWebSearch* webSearch)
+{
+ webSearch_editEngine_dialog_new(FALSE, webSearch);
+}
+
+static void on_webSearch_remove(GtkWidget* widget, CWebSearch* webSearch)
+{
+ GtkTreeSelection* selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(webSearch->treeview));
+ GtkTreeModel* liststore;
+ GtkTreeIter iter;
+ gtk_tree_selection_get_selected(selection, &liststore, &iter);
+ SearchEngine* searchEngine;
+ gtk_tree_model_get(liststore, &iter, ENGINES_COL_ENGINE, &searchEngine, -1);
+ gtk_list_store_remove(GTK_LIST_STORE(liststore), &iter);
+ search_engine_free(searchEngine);
+ searchEngines = g_list_remove(searchEngines, searchEngine);
+ update_searchEngine(config->searchEngine, webSearch->browser);
+ webSearch_toggle_edit_buttons(g_list_nth(searchEngines, 0) != NULL, webSearch);
+ // FIXME: we want to allow undo of some kind
+}
+
+GtkWidget* webSearch_manageSearchEngines_dialog_new(CBrowser* browser)
+{
+ const gchar* dialogTitle = "Manage search engines";
+ GtkWidget* dialog = gtk_dialog_new_with_buttons(dialogTitle
+ , GTK_WINDOW(browser->window)
+ , GTK_DIALOG_DESTROY_WITH_PARENT | GTK_DIALOG_NO_SEPARATOR
+ , GTK_STOCK_HELP
+ , GTK_RESPONSE_HELP
+ , GTK_STOCK_CLOSE
+ , GTK_RESPONSE_CLOSE
+ , NULL);
+ gtk_window_set_icon_name(GTK_WINDOW(dialog), GTK_STOCK_PROPERTIES);
+ // TODO: Implement some kind of help function
+ gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog)
+ , GTK_RESPONSE_HELP, FALSE); //...
+ gint iWidth, iHeight;
+ sokoke_widget_get_text_size(dialog, "M", &iWidth, &iHeight);
+ gtk_window_set_default_size(GTK_WINDOW(dialog), iWidth * 45, -1);
+ g_signal_connect(dialog, "response", G_CALLBACK(gtk_widget_destroy), dialog);
+ // TODO: Do we want tooltips for explainations or can we omit that?
+ // TODO: We need mnemonics
+ // TODO: Take multiple windows into account when applying changes
+ GtkWidget* xfce_heading;
+ if((xfce_heading = sokoke_xfce_header_new(
+ gtk_window_get_icon_name(GTK_WINDOW(dialog)), dialogTitle)))
+ gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), xfce_heading, FALSE, FALSE, 0);
+ GtkWidget* hbox = gtk_hbox_new(FALSE, 0);
+ gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, TRUE, TRUE, 12);
+ GtkTreeViewColumn* column;
+ GtkCellRenderer* renderer_text; GtkCellRenderer* renderer_pixbuf;
+ GtkListStore* liststore = gtk_list_store_new(ENGINES_COL_N
+ , G_TYPE_SEARCH_ENGINE);
+ GtkWidget* treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(liststore));
+ gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE);
+ column = gtk_tree_view_column_new();
+ renderer_pixbuf = gtk_cell_renderer_pixbuf_new();
+ gtk_tree_view_column_pack_start(column, renderer_pixbuf, FALSE);
+ gtk_tree_view_column_set_cell_data_func(column, renderer_pixbuf
+ , (GtkTreeCellDataFunc)on_webSearch_engines_render_icon, treeview, NULL);
+ renderer_text = gtk_cell_renderer_text_new();
+ gtk_tree_view_column_pack_start(column, renderer_text, TRUE);
+ gtk_tree_view_column_set_cell_data_func(column, renderer_text
+ , (GtkTreeCellDataFunc)on_webSearch_engines_render_text, treeview, NULL);
+ gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
+ GtkWidget* scrolled = gtk_scrolled_window_new(NULL, NULL);
+ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled)
+ , GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
+ gtk_container_add(GTK_CONTAINER(scrolled), treeview);
+ gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN);
+ gtk_box_pack_start(GTK_BOX(hbox), scrolled, TRUE, TRUE, 5);
+ guint n = g_list_length(searchEngines);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ SearchEngine* searchEngine = (SearchEngine*)g_list_nth_data(searchEngines, i);
+ gtk_list_store_insert_with_values(GTK_LIST_STORE(liststore), NULL, i
+ , ENGINES_COL_ENGINE, searchEngine, -1);
+ }
+ g_object_unref(liststore);
+ CWebSearch* webSearch = g_new0(CWebSearch, 1);
+ webSearch->browser = browser;
+ webSearch->window = dialog;
+ webSearch->treeview = treeview;
+ g_signal_connect(dialog, "response", G_CALLBACK(g_free), webSearch);
+ GtkWidget* vbox = gtk_vbox_new(FALSE, 4);
+ gtk_box_pack_start(GTK_BOX(hbox), vbox, FALSE, FALSE, 4);
+ GtkWidget* button = gtk_button_new_from_stock(GTK_STOCK_ADD);
+ g_signal_connect(button, "clicked", G_CALLBACK(on_webSearch_add), webSearch);
+ gtk_box_pack_start(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ button = gtk_button_new_from_stock(GTK_STOCK_EDIT);
+ g_signal_connect(button, "clicked", G_CALLBACK(on_webSearch_edit), webSearch);
+ gtk_box_pack_start(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ webSearch->edit = button;
+ button = gtk_button_new_from_stock(GTK_STOCK_REMOVE);
+ g_signal_connect(button, "clicked", G_CALLBACK(on_webSearch_remove), webSearch);
+ gtk_box_pack_start(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ webSearch->remove = button;
+ button = gtk_label_new(""); // This is an invisible separator
+ gtk_box_pack_start(GTK_BOX(vbox), button, TRUE, TRUE, 12);
+ button = gtk_button_new_from_stock(GTK_STOCK_GO_DOWN);
+ gtk_widget_set_sensitive(button, FALSE); //...
+ gtk_box_pack_end(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ button = gtk_button_new_from_stock(GTK_STOCK_GO_UP);
+ gtk_widget_set_sensitive(button, FALSE); //...
+ gtk_box_pack_end(GTK_BOX(vbox), button, FALSE, FALSE, 0);
+ webSearch_toggle_edit_buttons(n > 0, webSearch);
+ gtk_widget_show_all(GTK_DIALOG(dialog)->vbox);
+ return dialog;
+}
+
+gboolean on_webSearch_key_down(GtkWidget* widget, GdkEventKey* event, CBrowser* browser)
+{
+ GdkModifierType state = (GdkModifierType)0;
+ gint x, y; gdk_window_get_pointer(NULL, &x, &y, &state);
+ if(!(state & GDK_CONTROL_MASK))
+ return FALSE;
+ switch(event->keyval)
+ {
+ case GDK_Up:
+ update_searchEngine(config->searchEngine - 1, browser);
+ return TRUE;
+ case GDK_Down:
+ update_searchEngine(config->searchEngine + 1, browser);
+ return TRUE;
+ }
+ return FALSE;
+}
+
+gboolean on_webSearch_scroll(GtkWidget* webView, GdkEventScroll* event, CBrowser* browser)
+{
+ if(event->direction == GDK_SCROLL_DOWN)
+ update_searchEngine(config->searchEngine + 1, browser);
+ else if(event->direction == GDK_SCROLL_UP)
+ update_searchEngine(config->searchEngine - 1, browser);
+ return TRUE;
+}
+
+void on_webSearch_activate(GtkWidget* widget, CBrowser* browser)
+{
+ const gchar* keywords = gtk_entry_get_text(GTK_ENTRY(widget));
+ gchar* url;
+ SearchEngine* searchEngine = (SearchEngine*)g_list_nth_data(searchEngines, config->searchEngine);
+ if(searchEngine)
+ url = searchEngine->url;
+ else // The location search is our fallback
+ url = config->locationSearch;
+ gchar* search;
+ if(strstr(url, "%s"))
+ search = g_strdup_printf(url, keywords);
+ else
+ search = g_strconcat(url, " ", keywords, NULL);
+ entry_completion_append(GTK_ENTRY(widget), keywords);
+ webkit_web_view_open(WEBKIT_WEB_VIEW(get_nth_webView(-1, browser)), search);
+ g_free(search);
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __WEBSEARCH_H__
+#define __WEBSEARCH_H__ 1
+
+#include "browser.h"
+
+#include <gtk/gtk.h>
+#include <libsexy/sexy.h>
+#include <webkit.h>
+
+// -- Types
+
+typedef struct
+{
+ CBrowser* browser;
+ GtkWidget* window;
+ GtkWidget* treeview;
+ GtkWidget* edit;
+ GtkWidget* remove;
+} CWebSearch;
+
+enum
+{
+ ENGINES_COL_ENGINE,
+ ENGINES_COL_N
+};
+
+// -- Declarations
+
+void
+update_searchEngine(guint, CBrowser*);
+
+void
+on_webSearch_icon_released(GtkWidget*, SexyIconEntryPosition*, gint, CBrowser*);
+
+void
+on_webSearch_engine_activate(GtkWidget*, CBrowser*);
+
+void
+on_webSearch_activate(GtkWidget*, CBrowser*);
+
+GtkWidget*
+webSearch_manageSearchEngines_dialog_new(CBrowser*);
+
+gboolean
+on_webSearch_key_down(GtkWidget*, GdkEventKey*, CBrowser*);
+
+gboolean
+on_webSearch_scroll(GtkWidget*, GdkEventScroll*, CBrowser*);
+
+#endif /* !__WEBSEARCH_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "webView.h"
+
+#include "helpers.h"
+#include "sokoke.h"
+#include "xbel.h"
+
+#include <string.h>
+
+WebKitNavigationResponse on_webView_navigation_requested(GtkWidget* webView
+ , WebKitWebFrame* frame, WebKitNetworkRequest* networkRequest)
+{
+ WebKitNavigationResponse response = WEBKIT_NAVIGATION_RESPONSE_ACCEPT;
+ // TODO: Ask webkit wether it knows the protocol for "unknown protcol"
+ // TODO: This isn't the place for uri scheme handling
+ const gchar* uri = webkit_network_request_get_uri(networkRequest);
+ gchar* protocol = strtok(g_strdup(uri), ":");
+ if(spawn_protocol_command(protocol, uri))
+ response = WEBKIT_NAVIGATION_RESPONSE_IGNORE;
+ g_free(protocol);
+ return response;
+}
+
+void on_webView_location_changed(GtkWidget* webView, const gchar* uri
+, CBrowser* browser)
+{
+ gchar* newUri = g_strdup(uri ? uri : "");
+ xbel_bookmark_set_href(browser->sessionItem, newUri);
+ if(webView == get_nth_webView(-1, browser))
+ {
+ gtk_entry_set_text(GTK_ENTRY(browser->location), newUri);
+ gtk_label_set_text(GTK_LABEL(browser->webView_name), newUri);
+ update_status_message(NULL, browser);
+ update_gui_state(browser);
+ }
+}
+
+void on_webView_title_changed(GtkWidget* webView, const gchar* title
+ , const gchar* uri, CBrowser* browser)
+{
+ // TODO: We emulate location_changed here for now
+ // Shouldn't we have separated title_changed and location_changed signals?
+ on_webView_location_changed(webView, uri, browser);
+ gchar* newTitle;
+ newTitle = g_strdup(title ? title : uri);
+ xbel_item_set_title(browser->sessionItem, newTitle);
+ gtk_label_set_text(GTK_LABEL(browser->webView_name), newTitle);
+ sokoke_widget_set_tooltip_text(gtk_widget_get_parent(
+ gtk_widget_get_parent(browser->webView_name)), newTitle);
+ gtk_label_set_text(GTK_LABEL(gtk_bin_get_child(GTK_BIN(
+ browser->webView_menu))), newTitle);
+ if(webView == get_nth_webView(-1, browser))
+ {
+ gchar* windowTitle = g_strconcat(newTitle, " - ", PACKAGE_NAME, NULL);
+ gtk_window_set_title(GTK_WINDOW(browser->window), windowTitle);
+ g_free(windowTitle);
+ }
+}
+
+void on_webView_icon_changed(GtkWidget* webView, WebKitWebFrame* widget
+ , CBrowser* browser)
+{
+ // TODO: Implement icon updates; currently this isn't ever called anyway
+ const gchar* icon = NULL;
+ UNIMPLEMENTED
+ if(icon)
+ {
+ gtk_label_set_text(GTK_LABEL(browser->webView_name), "icon");
+ gtk_label_set_text(GTK_LABEL(gtk_bin_get_child(GTK_BIN(
+ browser->webView_menu))), "icon");
+ if(webView == get_nth_webView(-1, browser))
+ {
+ gchar* windowTitle = g_strconcat("icon", " - ", PACKAGE_NAME, NULL);
+ gtk_window_set_title(GTK_WINDOW(browser->window), windowTitle);
+ g_free(windowTitle);
+ }
+ }
+}
+
+void on_webView_load_started(GtkWidget* webView, WebKitWebFrame* widget
+ , CBrowser* browser)
+{
+ browser->loadedPercent = 0;
+ browser->loadedBytes = 0;
+ browser->loadedBytesMax = 0;
+ update_favicon(browser);
+ if(webView == get_nth_webView(-1, browser))
+ update_gui_state(browser);
+ update_statusbar_text(browser);
+ gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(browser->progress), 0.1);
+ gtk_widget_show(browser->progress);
+}
+
+void on_webView_load_changed(GtkWidget* webView, gint progress, CBrowser* browser)
+{
+ browser->loadedBytes = progress;
+ browser->loadedBytes = 100;
+ if(webView == get_nth_webView(-1, browser))
+ update_gui_state(browser);
+}
+
+void on_webView_load_finished(GtkWidget* webView, WebKitWebFrame* widget
+ , CBrowser* browser)
+{
+ browser->loadedPercent = -1;
+ update_favicon(browser);
+ if(webView == get_nth_webView(-1, browser))
+ update_gui_state(browser);
+ update_statusbar_text(browser);
+ gtk_widget_hide(browser->progress);
+}
+
+void on_webView_status_message(GtkWidget* webView, const gchar* text, CBrowser* browser)
+{
+ update_status_message(text, browser);
+}
+
+void on_webView_selection_changed(GtkWidget* webView, CBrowser* browser)
+{
+ UNIMPLEMENTED
+}
+
+gboolean on_webView_console_message(GtkWidget* webView
+ , const gchar* message, gint line, const gchar* sourceId, CBrowser* browser)
+{
+ return FALSE;
+}
+
+void on_webView_link_hover(GtkWidget* webView, const gchar* tooltip
+ , const gchar* uri, CBrowser* browser)
+{
+ update_status_message(uri, browser);
+ g_free(browser->elementUri);
+ browser->elementUri = g_strdup(uri);
+}
+
+/*
+GtkWidget* on_webView_window_open(GtkWidget* webView, const gchar* sUri
+ , CBrowser* browser)
+{
+ // A window is created
+ // TODO: Respect config->iNewPages
+ // TODO: Find out if this comes from a script or a click
+ // TODO: Block scripted popups, return NULL and show status icon
+ CBrowser* newBrowser = browser_new(config->openPopupsInTabs ? browser : NULL);
+ return newBrowser->webView;
+}
+*/
+
+void webView_popup(GtkWidget* webView, GdkEventButton* event, CBrowser* browser)
+{
+ gboolean isLink = browser->elementUri != NULL; // Did we right-click a link?
+ gboolean haveLinks = FALSE; // TODO: Are several links selected?
+ gboolean isImage = FALSE; // TODO: Did we right-click an image?
+ gboolean isEditable = FALSE; //webkit_web_view_can_paste_clipboard(WEBKIT_WEB_VIEW(webView)); //...
+ gchar* selection = NULL; //webkit_web_view_get_selected_text(WEBKIT_WEB_VIEW(webView));
+
+ update_edit_items(browser);
+
+ // Download manager available?
+ const gchar* downloadManager = g_datalist_get_data(&config->protocols_commands, "download");
+ gboolean canDownload = downloadManager && *downloadManager;
+
+ action_set_visible("LinkTabNew", isLink, browser);
+ action_set_visible("LinkTabCurrent", isLink, browser);
+ action_set_visible("LinkWindowNew", isLink, browser);
+
+ action_set_visible("LinkSaveAs", isLink, browser);
+ action_set_visible("LinkSaveWith", isLink && canDownload, browser);
+ action_set_visible("LinkCopy", isLink, browser);
+ action_set_visible("LinkBookmarkNew", isLink, browser);
+
+ action_set_visible("SelectionLinksNewTabs", haveLinks && selection && *selection, browser);
+ action_set_visible("SelectionTextTabNew", haveLinks && selection && *selection, browser);
+ action_set_visible("SelectionTextTabCurrent", haveLinks && selection && *selection, browser);
+ action_set_visible("SelectionTextWindowNew", haveLinks && selection && *selection, browser);
+
+ action_set_visible("ImageViewTabNew", isImage, browser);
+ action_set_visible("ImageViewTabCurrent", isImage, browser);
+ action_set_visible("ImageSaveAs", isImage, browser);
+ action_set_visible("ImageSaveWith", isImage && canDownload, browser);
+ action_set_visible("ImageCopy", isImage, browser);
+
+ action_set_visible("Copy_", (selection && *selection) || isEditable, browser);
+ action_set_visible("SelectionSearch", selection && *selection, browser);
+ action_set_visible("SelectionSearchWith", selection && *selection, browser);
+ action_set_visible("SelectionSourceView", selection && *selection, browser);
+
+ action_set_visible("SourceView", !selection, browser);
+
+ if(isEditable)
+ sokoke_widget_popup(webView, GTK_MENU(browser->popup_editable), event);
+ else if(isLink || isImage || (selection && *selection))
+ sokoke_widget_popup(webView, GTK_MENU(browser->popup_element), event);
+ else
+ sokoke_widget_popup(webView, GTK_MENU(browser->popup_webView), event);
+}
+
+gboolean on_webView_button_release(GtkWidget* webView, GdkEventButton* event
+ , CBrowser* browser)
+{
+ GdkModifierType state = (GdkModifierType)0;
+ gint x, y;
+ gdk_window_get_pointer(NULL, &x, &y, &state);
+ switch(event->button)
+ {
+ case 1:
+ if(!browser->elementUri) return FALSE;
+ if(state & GDK_SHIFT_MASK)
+ {
+ // Open link in new window
+ CBrowser* curBrowser = browser_new(NULL);
+ webkit_web_view_open(WEBKIT_WEB_VIEW(curBrowser->webView), browser->elementUri);
+ return TRUE;
+ }
+ else if(state & GDK_MOD1_MASK)
+ {
+ // Open link in new tab
+ CBrowser* curBrowser = browser_new(browser);
+ webkit_web_view_open(WEBKIT_WEB_VIEW(curBrowser->webView), browser->elementUri);
+ return TRUE;
+ }
+ break;
+ case 2:
+ if(state & GDK_CONTROL_MASK)
+ {
+ //webkit_web_view_set_text_size(WEBKIT_WEB_VIEW(webView), 1);
+ return TRUE;
+ }
+ else
+ {
+ if(!browser->elementUri) return FALSE;
+ // Open link in new tab
+ CBrowser* curBrowser = browser_new(browser);
+ webkit_web_view_open(WEBKIT_WEB_VIEW(curBrowser->webView), browser->elementUri);
+ return TRUE;
+ }
+ break;
+ case 3:
+ webView_popup(webView, event, browser);
+ return TRUE;
+ }
+ return FALSE;
+}
+
+void on_webView_popup(GtkWidget* webView, CBrowser* browser)
+{
+ webView_popup(webView, NULL, browser);
+}
+
+gboolean on_webView_scroll(GtkWidget* webView, GdkEventScroll* event
+ , CBrowser* browser)
+{
+ GdkModifierType state = (GdkModifierType)0;
+ gint x, y;
+ gdk_window_get_pointer(NULL, &x, &y, &state);
+ if(state & GDK_CONTROL_MASK)
+ {
+ /*const gfloat size = webkit_web_view_get_text_size(WEBKIT_WEB_VIEW(webView));
+ if(event->direction == GDK_SCROLL_DOWN)
+ webkit_web_view_set_text_size(WEBKIT_WEB_VIEW(webView), size + 0.1);
+ else if(event->direction == GDK_SCROLL_UP)
+ webView_set_text_size(WEBKIT_WEB_VIEW(webView), size - 0.1);*/
+ return TRUE;
+ }
+ else
+ return FALSE;
+}
+
+gboolean on_webView_leave(GtkWidget* webView, GdkEventCrossing* event, CBrowser* browser)
+{
+ update_status_message(NULL, browser);
+ return TRUE;
+}
+
+void on_webView_destroy(GtkWidget* widget, CBrowser* browser)
+{
+ // Update browser list, free memory and possibly quit
+ GList* tmp = g_list_find(browsers, browser);
+ browsers = g_list_delete_link(browsers, tmp);
+ g_free(browser->elementUri);
+ g_free(browser->statusMessage);
+ if(!g_list_length(browsers))
+ {
+ g_object_unref(browser->actiongroup);
+ g_object_unref(browser->popup_webView);
+ g_object_unref(browser->popup_element);
+ g_object_unref(browser->popup_editable);
+ gtk_main_quit();
+ }
+}
+
+// webView actions begin here
+
+GtkWidget* webView_new(GtkWidget** scrolled)
+{
+ *scrolled = gtk_scrolled_window_new(NULL, NULL);
+ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(*scrolled)
+ , GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
+ GTK_WIDGET_SET_FLAGS(*scrolled, GTK_CAN_FOCUS);
+ GtkWidget* webView = webkit_web_view_new();
+ gtk_container_add(GTK_CONTAINER(*scrolled), webView);
+ return webView;
+}
+
+void webView_open(GtkWidget* webView, const gchar* uri)
+{
+ webkit_web_view_open(WEBKIT_WEB_VIEW(webView), (gchar*)uri);
+ // We need to check the browser first
+ // No browser means this is a panel
+ CBrowser* browser = get_browser_from_webView(webView);
+ if(browser)
+ {
+ xbel_bookmark_set_href(browser->sessionItem, uri);
+ xbel_item_set_title(browser->sessionItem, "");
+ }
+}
+
+void webView_close(GtkWidget* webView, CBrowser* browser)
+{
+ browser = get_browser_from_webView(webView);
+ const gchar* uri = xbel_bookmark_get_href(browser->sessionItem);
+ xbel_folder_remove_item(session, browser->sessionItem);
+ if(uri && *uri)
+ {
+ xbel_folder_prepend_item(tabtrash, browser->sessionItem);
+ guint n = xbel_folder_get_n_items(tabtrash);
+ if(n > 10)
+ {
+ XbelItem* item = xbel_folder_get_nth_item(tabtrash, n - 1);
+ xbel_folder_remove_item(tabtrash, item);
+ xbel_item_free(item);
+ }
+ }
+ else
+ xbel_item_free(browser->sessionItem);
+ gtk_widget_destroy(browser->webView_menu);
+ gtk_notebook_remove_page(GTK_NOTEBOOK(browser->webViews)
+ , get_webView_index(webView, browser));
+ update_browser_actions(browser);
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __webView_H__
+#define __webView_H__ 1
+
+#include <gtk/gtk.h>
+#include "browser.h"
+#include "debug.h"
+
+#include <webkit.h>
+
+WebKitNavigationResponse
+on_webView_navigation_requested(GtkWidget* webView, WebKitWebFrame* frame
+ , WebKitNetworkRequest* networkRequest);
+
+void
+on_webView_location_changed(GtkWidget*, const gchar*, CBrowser*);
+
+void
+on_webView_title_changed(GtkWidget*, const gchar*, const gchar*, CBrowser*);
+
+void
+on_webView_icon_changed(GtkWidget*, WebKitWebFrame*, CBrowser*);
+
+void
+on_webView_load_started(GtkWidget* , WebKitWebFrame*, CBrowser*);
+
+void
+on_webView_load_changed(GtkWidget*, gint progress, CBrowser*);
+
+void
+on_webView_load_finished(GtkWidget*, WebKitWebFrame*, CBrowser*);
+
+void
+on_webView_status_message(GtkWidget*, const gchar*, CBrowser*);
+
+void
+on_webView_selection_changed(GtkWidget*, CBrowser*);
+
+gboolean
+on_webView_console_message(GtkWidget*, const gchar*, gint, const gchar*, CBrowser*);
+
+void
+on_webView_link_hover(GtkWidget*, const gchar*, const gchar*, CBrowser*);
+
+/*
+GtkWidget*
+on_webView_window_open(GtkWidget*, const gchar*, CBrowser*);
+*/
+
+gboolean
+on_webView_button_release(GtkWidget*, GdkEventButton*, CBrowser*);
+
+void
+on_webView_popup(GtkWidget*, CBrowser*);
+
+gboolean
+on_webView_scroll(GtkWidget*, GdkEventScroll*, CBrowser*);
+
+gboolean
+on_webView_leave(GtkWidget*, GdkEventCrossing*, CBrowser*);
+
+void
+on_webView_destroy(GtkWidget*, CBrowser*);
+
+GtkWidget*
+webView_new(GtkWidget**);
+
+void
+webView_open(GtkWidget*, const gchar*);
+
+void
+webView_close(GtkWidget*, CBrowser*);
+
+#endif /* !__webView_H__ */
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+/**
+ * This is an implementation of XBEL based on Glib and libXML2.
+ *
+ * Design Goals:
+ * - XbelItem is the only opaque public data structure.
+ * - The interface should be intuitive and familiar to Glib users.
+ * - There should be no public exposure of libXML2 specific code.
+ * - Bookmarks should actually be easily exchangeable between programs.
+ *
+ * TODO:
+ * - Support info > metadata, alias, added, modified, visited
+ * - Compatibility: The Nokia 770 *requires* metadata and folder
+ * - Compatibility: Kazehakase's bookmarks
+ * - Compatibility: Epiphany's bookmarks
+ * - XML Indentation
+ * - Export and import to other formats
+ **/
+
+#include "xbel.h"
+
+#include <stdio.h>
+#include <string.h>
+#include <libxml/parser.h>
+#include <libxml/tree.h>
+
+// Private: Create a new item of a certain type
+static XbelItem* xbel_item_new(XbelItemKind kind)
+{
+ XbelItem* item = g_new(XbelItem, 1);
+ item->parent = NULL;
+ item->kind = kind;
+ if(kind == XBEL_ITEM_FOLDER)
+ {
+ item->items = NULL;
+ item->folded = TRUE;
+ }
+ if(kind != XBEL_ITEM_SEPARATOR)
+ {
+ item->title = NULL;
+ item->desc = NULL;
+ }
+ if(kind == XBEL_ITEM_BOOKMARK)
+ item->href = g_strdup("");
+ return item;
+}
+
+/**
+ * xbel_bookmark_new:
+ *
+ * Create a new empty bookmark.
+ *
+ * Return value: a newly allocated bookmark
+ **/
+XbelItem* xbel_bookmark_new(void)
+{
+ return xbel_item_new(XBEL_ITEM_BOOKMARK);
+}
+
+static XbelItem* xbel_bookmark_from_xmlNodePtr(xmlNodePtr cur)
+{
+ g_return_val_if_fail(cur != NULL, NULL);
+ XbelItem* bookmark = xbel_bookmark_new();
+ xmlChar* key = xmlGetProp(cur, (xmlChar*)"href");
+ xbel_bookmark_set_href(bookmark, (gchar*)key);
+ cur = cur->xmlChildrenNode;
+ while(cur != NULL)
+ {
+ if(!xmlStrcmp(cur->name, (const xmlChar*)"title"))
+ {
+ xmlChar* key = xmlNodeGetContent(cur);
+ bookmark->title = (gchar*)g_strstrip((gchar*)key);
+ }
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"desc"))
+ {
+ xmlChar* key = xmlNodeGetContent(cur);
+ bookmark->desc = (gchar*)g_strstrip((gchar*)key);
+ }
+ cur = cur->next;
+ }
+ return bookmark;
+}
+
+/**
+ * xbel_separator_new:
+ *
+ * Create a new separator.
+ *
+ * The returned item must be freed eventually.
+ *
+ * Return value: a newly allocated separator.
+ **/
+XbelItem* xbel_separator_new(void)
+{
+ return xbel_item_new(XBEL_ITEM_SEPARATOR);
+}
+
+/**
+ * xbel_folder_new:
+ *
+ * Create a new empty folder.
+ *
+ * The returned item must be freed eventually.
+ *
+ * Return value: a newly allocated folder.
+ **/
+XbelItem* xbel_folder_new(void)
+{
+ return xbel_item_new(XBEL_ITEM_FOLDER);
+}
+
+// Private: Create a folder from an xmlNodePtr
+static XbelItem* xbel_folder_from_xmlNodePtr(xmlNodePtr cur)
+{
+ g_return_val_if_fail(cur != NULL, NULL);
+ XbelItem* folder = xbel_folder_new();
+ xmlChar* key = xmlGetProp(cur, (xmlChar*)"folded");
+ if(key)
+ {
+ if(!g_ascii_strncasecmp((gchar*)key, "yes", 3))
+ folder->folded = TRUE;
+ else if(!g_ascii_strncasecmp((gchar*)key, "no", 2))
+ folder->folded = FALSE;
+ else
+ g_warning("XBEL: Unknown value for folded.");
+ xmlFree(key);
+ }
+ cur = cur->xmlChildrenNode;
+ while(cur)
+ {
+ if(!xmlStrcmp(cur->name, (const xmlChar*)"title"))
+ {
+ xmlChar* key = xmlNodeGetContent(cur);
+ folder->title = (gchar*)g_strstrip((gchar*)key);
+ }
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"desc"))
+ {
+ xmlChar* key = xmlNodeGetContent(cur);
+ folder->desc = (gchar*)g_strstrip((gchar*)key);
+ }
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"folder"))
+ {
+ XbelItem* item = xbel_folder_from_xmlNodePtr(cur);
+ item->parent = (struct XbelItem*)folder;
+ folder->items = g_list_prepend(folder->items, item);
+ }
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"bookmark"))
+ {
+ XbelItem* item = xbel_bookmark_from_xmlNodePtr(cur);
+ item->parent = (struct XbelItem*)folder;
+ folder->items = g_list_prepend(folder->items, item);
+ }
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"separator"))
+ {
+ XbelItem* item = xbel_separator_new();
+ item->parent = (struct XbelItem*)folder;
+ folder->items = g_list_prepend(folder->items, item);
+ }
+ cur = cur->next;
+ }
+ // Prepending and reversing is faster than appending
+ folder->items = g_list_reverse(folder->items);
+ return folder;
+}
+
+// Private: Loads the contents from an xmlNodePtr into a folder.
+static gboolean xbel_folder_from_xmlDocPtr(XbelItem* folder, xmlDocPtr doc)
+{
+ g_return_val_if_fail(xbel_folder_is_empty(folder), FALSE);
+ g_return_val_if_fail(doc != NULL, FALSE);
+ xmlNodePtr cur = xmlDocGetRootElement(doc);
+ xmlChar* version = xmlGetProp(cur, (xmlChar*)"version");
+ if(xmlStrcmp(version, (xmlChar*)"1.0"))
+ g_warning("XBEL version is not 1.0.");
+ xmlFree(version);
+ folder->title = (gchar*)xmlGetProp(cur, (xmlChar*)"title");
+ folder->desc = (gchar*)xmlGetProp(cur, (xmlChar*)"desc");
+ if((cur = xmlDocGetRootElement(doc)) == NULL)
+ {
+ // Empty document
+ return FALSE;
+ }
+ if(xmlStrcmp(cur->name, (const xmlChar*)"xbel"))
+ {
+ // Wrong document kind
+ return FALSE;
+ }
+ cur = cur->xmlChildrenNode;
+ while(cur != NULL)
+ {
+ XbelItem* item = NULL;
+ if(!xmlStrcmp(cur->name, (const xmlChar*)"folder"))
+ item = xbel_folder_from_xmlNodePtr(cur);
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"bookmark"))
+ item = xbel_bookmark_from_xmlNodePtr(cur);
+ else if(!xmlStrcmp(cur->name, (const xmlChar*)"separator"))
+ item = xbel_separator_new();
+ /*else if(!xmlStrcmp(cur->name, (const xmlChar*)"info"))
+ item = xbel_parse_info(xbel, cur);*/
+ if(item != NULL)
+ {
+ item->parent = (struct XbelItem*)folder;
+ folder->items = g_list_prepend(folder->items, item);
+ }
+ cur = cur->next;
+ }
+ // Prepending and reversing is faster than appending
+ folder->items = g_list_reverse(folder->items);
+ return TRUE;
+}
+
+/**
+ * xbel_item_free:
+ * @item: a valid item
+ *
+ * Free an XbelItem. If @item is a folder all of its children will also
+ * be freed automatically.
+ *
+ * The item must not be contained in a folder or attempting to free it will fail.
+ **/
+void xbel_item_free(XbelItem* item)
+{
+ g_return_if_fail(item);
+ g_return_if_fail(!xbel_item_get_parent(item));
+ if(xbel_item_is_folder(item))
+ {
+ guint n = xbel_folder_get_n_items(item);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ XbelItem* _item = xbel_folder_get_nth_item(item, i);
+ _item->parent = NULL;
+ xbel_item_free(_item);
+ }
+ g_list_free(item->items);
+ }
+ if(item->kind != XBEL_ITEM_SEPARATOR)
+ {
+ g_free(item->title);
+ g_free(item->desc);
+ }
+ if(item->kind == XBEL_ITEM_BOOKMARK)
+ g_free(item->href);
+ g_free(item);
+}
+
+/**
+ * xbel_item_copy:
+ * @item: the item to copy
+ *
+ * Copy an XbelItem.
+ *
+ * The returned item must be freed eventually.
+ *
+ * Return value: a copy of @item
+ **/
+XbelItem* xbel_item_copy(XbelItem* item)
+{
+ g_return_val_if_fail(item, NULL);
+ XbelItem* copy = xbel_folder_new();
+ if(xbel_item_is_folder(item))
+ {
+ guint n = xbel_folder_get_n_items(item);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ XbelItem* _item = xbel_folder_get_nth_item(item, i);
+ xbel_folder_append_item(copy, xbel_item_copy(_item));
+ }
+ }
+ if(item->kind != XBEL_ITEM_SEPARATOR)
+ {
+ xbel_item_set_title(copy, item->title);
+ xbel_item_set_desc(copy, item->desc);
+ }
+ if(item->kind == XBEL_ITEM_BOOKMARK)
+ xbel_bookmark_set_href(copy, item->href);
+ return copy;
+}
+
+GType xbel_item_get_type()
+{
+ static GType type = 0;
+ if(!type)
+ type = g_pointer_type_register_static("xbel_item");
+ return type;
+}
+
+/**
+ * xbel_folder_append_item:
+ * @folder: a folder
+ * @item: the item to append
+ *
+ * Append the given item to a folder.
+ *
+ * The item is actually moved and must not be contained in another folder.
+ *
+ **/
+void xbel_folder_append_item(XbelItem* folder, XbelItem* item)
+{
+ g_return_if_fail(!xbel_item_get_parent(item));
+ g_return_if_fail(xbel_item_is_folder(folder));
+ item->parent = (struct XbelItem*)folder;
+ folder->items = g_list_append(folder->items, item);
+}
+
+/**
+ * xbel_folder_prepend_item:
+ * @folder: a folder
+ * @item: the item to prepend
+ *
+ * Prepend the given item to a folder.
+ *
+ * The item is actually moved and must not be contained in another folder.
+ *
+ **/
+void xbel_folder_prepend_item(XbelItem* folder, XbelItem* item)
+{
+ g_return_if_fail(!xbel_item_get_parent(item));
+ g_return_if_fail(xbel_item_is_folder(folder));
+ item->parent = (struct XbelItem*)folder;
+ folder->items = g_list_prepend(folder->items, item);
+}
+
+/**
+ * xbel_folder_remove_item:
+ * @folder: a folder
+ * @item: the item to remove
+ *
+ * Remove the given @item from a @folder.
+ **/
+void xbel_folder_remove_item(XbelItem* folder, XbelItem* item)
+{
+ g_return_if_fail(item);
+ g_return_if_fail(xbel_item_get_parent(folder) != item);
+ item->parent = NULL;
+ // Fortunately we know that items are unique
+ folder->items = g_list_remove(folder->items, item);
+}
+
+/**
+ * xbel_folder_get_n_items:
+ * @folder: a folder
+ *
+ * Retrieve the number of items contained in the given @folder.
+ *
+ * Return value: number of items
+ **/
+guint xbel_folder_get_n_items(XbelItem* folder)
+{
+ g_return_val_if_fail(xbel_item_is_folder(folder), 0);
+ return g_list_length(folder->items);
+}
+
+/**
+ * xbel_folder_get_nth_item:
+ * @folder: a folder
+ * @n: the index of an item
+ *
+ * Retrieve an item contained in the given @folder by its index.
+ *
+ * Return value: the item at the given index or %NULL
+ **/
+XbelItem* xbel_folder_get_nth_item(XbelItem* folder, guint n)
+{
+ g_return_val_if_fail(xbel_item_is_folder(folder), NULL);
+ return (XbelItem*)g_list_nth_data(folder->items, n);
+}
+
+/**
+ * xbel_folder_is_empty:
+ * @folder: A folder.
+ *
+ * Determines wether or not a folder contains no items. This is significantly
+ * faster than xbel_folder_get_n_items for this particular purpose.
+ *
+ * Return value: Wether the given @folder is folded.
+ **/
+gboolean xbel_folder_is_empty(XbelItem* folder)
+{
+ return !xbel_folder_get_nth_item(folder, 0);
+}
+
+/**
+ * xbel_folder_get_folded:
+ * @folder: A folder.
+ *
+ * Determines wether or not a folder is folded. If a folder is not folded
+ * it should not be exposed in a user interface.
+ *
+ * New folders are folded by default.
+ *
+ * Return value: Wether the given @folder is folded.
+ **/
+gboolean xbel_folder_get_folded(XbelItem* folder)
+{
+ g_return_val_if_fail(xbel_item_is_folder(folder), TRUE);
+ return folder->folded;
+}
+
+/**
+ * xbel_item_get_kind:
+ * @item: A item.
+ *
+ * Determines the kind of an item.
+ *
+ * Return value: The kind of the given @item.
+ **/
+XbelItemKind xbel_item_get_kind(XbelItem* item)
+{
+ return item->kind;
+}
+
+/**
+ * xbel_item_get_parent:
+ * @item: A valid item.
+ *
+ * Retrieves the parent folder of an item.
+ *
+ * Return value: The parent folder of the given @item or %NULL.
+ **/
+XbelItem* xbel_item_get_parent(XbelItem* item)
+{
+ g_return_val_if_fail(item, NULL);
+ return (XbelItem*)item->parent;
+}
+
+/**
+ * xbel_item_get_title:
+ * @item: A valid item.
+ *
+ * Retrieves the title of an item.
+ *
+ * Return value: The title of the given @item or %NULL.
+ **/
+G_CONST_RETURN gchar* xbel_item_get_title(XbelItem* item)
+{
+ g_return_val_if_fail(!xbel_item_is_separator(item), NULL);
+ return item->title;
+}
+
+/**
+ * xbel_item_get_desc:
+ * @item: A valid item.
+ *
+ * Retrieves the description of an item.
+ *
+ * Return value: The description of the @item or %NULL.
+ **/
+G_CONST_RETURN gchar* xbel_item_get_desc(XbelItem* item)
+{
+ g_return_val_if_fail(!xbel_item_is_separator(item), NULL);
+ return item->desc;
+}
+
+/**
+ * xbel_bookmark_get_href:
+ * @bookmark: A bookmark.
+ *
+ * Retrieves the uri of a bookmark. The value is guaranteed to not be %NULL.
+ *
+ * Return value: The uri of the @bookmark.
+ **/
+G_CONST_RETURN gchar* xbel_bookmark_get_href(XbelItem* bookmark)
+{
+ g_return_val_if_fail(xbel_item_is_bookmark(bookmark), NULL);
+ return bookmark->href;
+}
+
+/**
+ * xbel_item_is_bookmark:
+ * @item: A valid item.
+ *
+ * Determines wether or not an item is a bookmark.
+ *
+ * Return value: %TRUE if the @item is a bookmark.
+ **/
+gboolean xbel_item_is_bookmark(XbelItem* item)
+{
+ g_return_val_if_fail(item, FALSE);
+ return item->kind == XBEL_ITEM_BOOKMARK;
+}
+
+/**
+ * xbel_item_is_separator:
+ * @item: A valid item.
+ *
+ * Determines wether or not an item is a separator.
+ *
+ * Return value: %TRUE if the @item is a separator.
+ **/
+gboolean xbel_item_is_separator(XbelItem* item)
+{
+ g_return_val_if_fail(item, FALSE);
+ return item->kind == XBEL_ITEM_SEPARATOR;
+}
+
+/**
+ * xbel_item_is_folder:
+ * @item: A valid item.
+ *
+ * Determines wether or not an item is a folder.
+ *
+ * Return value: %TRUE if the item is a folder.
+ **/
+gboolean xbel_item_is_folder(XbelItem* item)
+{
+ g_return_val_if_fail(item, FALSE);
+ return item->kind == XBEL_ITEM_FOLDER;
+}
+
+/**
+ * xbel_folder_set_folded:
+ * @folder: A folder.
+ * @folded: TRUE if the folder is folded.
+ *
+ * Sets the foldedness of the @folder.
+ **/
+void xbel_folder_set_folded(XbelItem* folder, gboolean folded)
+{
+ g_return_if_fail(xbel_item_is_folder(folder));
+ folder->folded = folded;
+}
+
+/**
+ * xbel_item_set_title:
+ * @item: A valid item.
+ * @title: A string to use for the title.
+ *
+ * Sets the title of the @item.
+ **/
+void xbel_item_set_title(XbelItem* item, const gchar* title)
+{
+ g_return_if_fail(!xbel_item_is_separator(item));
+ g_free(item->title);
+ item->title = g_strdup(title);
+}
+
+/**
+ * xbel_item_set_desc:
+ * @item: A valid item.
+ * @title: A string to use for the description.
+ *
+ * Sets the description of the @item.
+ **/
+void xbel_item_set_desc(XbelItem* item, const gchar* desc)
+{
+ g_return_if_fail(!xbel_item_is_separator(item));
+ g_free(item->desc);
+ item->desc = g_strdup(desc);
+}
+
+/**
+ * xbel_bookmark_set_href:
+ * @bookmark: A bookmark.
+ * @href: A string containing a valid uri.
+ *
+ * Sets the uri of the bookmark.
+ *
+ * The uri must not be %NULL.
+ *
+ * This uri is not currently validated in any way. This may change in the future.
+ **/
+void xbel_bookmark_set_href(XbelItem* bookmark, const gchar* href)
+{
+ g_return_if_fail(xbel_item_is_bookmark(bookmark));
+ g_return_if_fail(href);
+ g_free(bookmark->href);
+ bookmark->href = g_strdup(href);
+}
+
+gboolean xbel_folder_from_data(XbelItem* folder, const gchar* data, GError** error)
+{
+ g_return_val_if_fail(xbel_folder_is_empty(folder), FALSE);
+ g_return_val_if_fail(data, FALSE);
+ xmlDocPtr doc;
+ if((doc = xmlParseMemory(data, strlen(data))) == NULL)
+ {
+ // No valid xml or broken encoding
+ *error = g_error_new(XBEL_ERROR, XBEL_ERROR_READ
+ , "Malformed document.");
+ return FALSE;
+ }
+ if(!xbel_folder_from_xmlDocPtr(folder, doc))
+ {
+ // Parsing failed
+ xmlFreeDoc(doc);
+ *error = g_error_new(XBEL_ERROR, XBEL_ERROR_READ
+ , "Malformed document.");
+ return FALSE;
+ }
+ xmlFreeDoc(doc);
+ return TRUE;
+}
+
+/**
+ * xbel_folder_from_file:
+ * @folder: An empty folder.
+ * @file: A relative path to a file.
+ * @error: return location for a GError or %NULL
+ *
+ * Tries to load @file.
+ *
+ * Return value: %TRUE on success or %FALSE when an error occured.
+ **/
+gboolean xbel_folder_from_file(XbelItem* folder, const gchar* file, GError** error)
+{
+ g_return_val_if_fail(xbel_folder_is_empty(folder), FALSE);
+ g_return_val_if_fail(file, FALSE);
+ if(!g_file_test(file, G_FILE_TEST_EXISTS))
+ {
+ // File doesn't exist
+ *error = g_error_new(G_FILE_ERROR, G_FILE_ERROR_NOENT
+ , "File not found.");
+ return FALSE;
+ }
+ xmlDocPtr doc;
+ if((doc = xmlParseFile(file)) == NULL)
+ {
+ // No valid xml or broken encoding
+ *error = g_error_new(XBEL_ERROR, XBEL_ERROR_READ
+ , "Malformed document.");
+ return FALSE;
+ }
+ if(!xbel_folder_from_xmlDocPtr(folder, doc))
+ {
+ // Parsing failed
+ xmlFreeDoc(doc);
+ *error = g_error_new(XBEL_ERROR, XBEL_ERROR_READ
+ , "Malformed document.");
+ return FALSE;
+ }
+ xmlFreeDoc(doc);
+ return TRUE;
+}
+
+/**
+ * xbel_folder_from_data_dirs:
+ * @folder: An empty folder.
+ * @file: A relative path to a file.
+ * @full_path: return location for the full path of the file or %NULL
+ * @error: return location for a GError or %NULL
+ *
+ * Tries to load @file from the user data dir or any of the system data dirs.
+ *
+ * Return value: %TRUE on success or %FALSE when an error occured.
+ **/
+gboolean xbel_folder_from_data_dirs(XbelItem* folder, const gchar* file
+ , gchar** full_path, GError** error)
+{
+ g_return_val_if_fail(xbel_folder_is_empty(folder), FALSE);
+ g_return_val_if_fail(file, FALSE);
+ // FIXME: Essentially unimplemented
+
+ *error = g_error_new(XBEL_ERROR, XBEL_ERROR_READ
+ , "Malformed document.");
+ return FALSE;
+}
+
+static gchar* xbel_xml_element(const gchar* name, const gchar* value)
+{
+ if(!value)
+ return g_strdup("");
+ gchar* valueEscaped = g_markup_escape_text(value, -1);
+ gchar* XML = g_strdup_printf("<%s>%s</%s>\n", name, valueEscaped, name);
+ g_free(valueEscaped);
+ return XML;
+}
+
+static gchar* xbel_item_to_data(XbelItem* item)
+{
+ g_return_val_if_fail(item, NULL);
+ gchar* XML = NULL;
+ switch(xbel_item_get_kind(item))
+ {
+ case XBEL_ITEM_FOLDER:
+ {
+ GString* _XML = g_string_new(NULL);
+ guint n = xbel_folder_get_n_items(item);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ XbelItem* _item = xbel_folder_get_nth_item(item, i);
+ gchar* itemXML = xbel_item_to_data(_item);
+ g_string_append(_XML, itemXML);
+ g_free(itemXML);
+ }
+ gchar* folded = item->folded ? NULL : g_strdup_printf(" folded=\"no\"");
+ gchar* title = xbel_xml_element("title", item->title);
+ gchar* desc = xbel_xml_element("desc", item->desc);
+ XML = g_strdup_printf("<folder%s>\n%s%s%s</folder>\n"
+ , folded ? folded : ""
+ , title
+ , desc
+ , g_string_free(_XML, FALSE));
+ g_free(folded);
+ g_free(title);
+ g_free(desc);
+ break;
+ }
+ case XBEL_ITEM_BOOKMARK:
+ {
+ gchar* hrefEscaped = g_markup_escape_text(item->href, -1);
+ gchar* href = g_strdup_printf(" href=\"%s\"", hrefEscaped);
+ g_free(hrefEscaped);
+ gchar* title = xbel_xml_element("title", item->title);
+ gchar* desc = xbel_xml_element("desc", item->desc);
+ XML = g_strdup_printf("<bookmark%s>\n%s%s%s</bookmark>\n"
+ , href
+ , title
+ , desc
+ , "");
+ g_free(href);
+ g_free(title);
+ g_free(desc);
+ break;
+ }
+ case XBEL_ITEM_SEPARATOR:
+ XML = g_strdup("<separator/>\n");
+ break;
+ default:
+ g_warning("XBEL: Unknown item kind");
+ }
+ return XML;
+}
+
+/**
+ * xbel_folder_to_data:
+ * @folder: A folder.
+ * @length: return location for the length of the created string or %NULL
+ * @error: return location for a GError or %NULL
+ *
+ * Retrieve the contents of @folder as a string.
+ *
+ * Return value: %TRUE on success or %FALSE when an error occured.
+ **/
+gchar* xbel_folder_to_data(XbelItem* folder, gsize* length, GError** error)
+{
+ g_return_val_if_fail(xbel_item_is_folder(folder), FALSE);
+ // FIXME: length is never filled
+ GString* innerXML = g_string_new(NULL);
+ guint n = xbel_folder_get_n_items(folder);
+ guint i;
+ for(i = 0; i < n; i++)
+ {
+ gchar* sItem = xbel_item_to_data(xbel_folder_get_nth_item(folder, i));
+ g_string_append(innerXML, sItem);
+ g_free(sItem);
+ }
+ gchar* title = xbel_xml_element("title", folder->title);
+ gchar* desc = xbel_xml_element("desc", folder->desc);
+ gchar* outerXML;
+ outerXML = g_strdup_printf("%s%s<xbel version=\"1.0\">\n%s%s%s</xbel>\n"
+ , "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+ , "<!DOCTYPE xbel PUBLIC \"+//IDN python.org//DTD XML Bookmark Exchange Language 1.0//EN//XML\" \"http://www.python.org/topics/xml/dtds/xbel-1.0.dtd\">\n"
+ , title
+ , desc
+ , g_string_free(innerXML, FALSE));
+ g_free(title);
+ g_free(desc);
+ return outerXML;
+}
+
+/**
+ * xbel_folder_to_file:
+ * @folder: A folder.
+ * @file: The destination filename.
+ * @error: return location for a GError or %NULL
+ *
+ * Write the contents of @folder to the specified file, creating it if necessary.
+ *
+ * Return value: %TRUE on success or %FALSE when an error occured.
+ **/
+gboolean xbel_folder_to_file(XbelItem* folder, const gchar* file, GError** error)
+{
+ g_return_val_if_fail(file, FALSE);
+ gchar* data;
+ if(!(data = xbel_folder_to_data(folder, NULL, error)))
+ return FALSE;
+ FILE* fp;
+ if(!(fp = fopen(file, "w")))
+ {
+ *error = g_error_new(G_FILE_ERROR, G_FILE_ERROR_ACCES
+ , "Writing failed.");
+ return FALSE;
+ }
+ fputs(data, fp);
+ fclose(fp);
+ g_free(data);
+ return TRUE;
+}
--- /dev/null
+/*
+ Copyright (C) 2007 Christian Dywan <christian@twotoasts.de>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __XBEL_H__
+#define __XBEL_H__ 1
+
+#include <glib.h>
+#include <glib-object.h>
+
+#define XBEL_ERROR g_quark_from_string("XBEL_ERROR")
+
+typedef enum
+{
+ XBEL_ERROR_INVALID_URI, /* Malformed uri */
+ XBEL_ERROR_INVALID_VALUE, /* Requested field not found */
+ XBEL_ERROR_URI_NOT_FOUND, /* Requested uri not found */
+ XBEL_ERROR_READ, /* Malformed document */
+ XBEL_ERROR_UNKNOWN_ENCODING, /* Parsed text was in an unknown encoding */
+ XBEL_ERROR_WRITE, /* Writing failed. */
+} XBELError;
+
+typedef enum
+{
+ XBEL_ITEM_FOLDER,
+ XBEL_ITEM_BOOKMARK,
+ XBEL_ITEM_SEPARATOR
+} XbelItemKind;
+
+// Note: This structure is entirely private.
+typedef struct
+{
+ XbelItemKind kind;
+ struct XbelItem* parent;
+
+ GList* items; // folder
+ gboolean folded; // foolder
+ gchar* title; // !separator
+ gchar* desc; // folder and bookmark
+ gchar* href; // bookmark
+ //time_t added; // !separator
+ //time_t modfied; // bookmark
+ //time_t visited; // bookmark
+} XbelItem;
+
+XbelItem*
+xbel_bookmark_new(void);
+
+XbelItem*
+xbel_separator_new(void);
+
+XbelItem*
+xbel_folder_new(void);
+
+void
+xbel_item_free(XbelItem*);
+
+XbelItem*
+xbel_item_copy(XbelItem*);
+
+GType
+xbel_item_get_type();
+
+#define G_TYPE_XBEL_ITEM xbel_item_get_type()
+
+void
+xbel_folder_append_item(XbelItem*, XbelItem*);
+
+void
+xbel_folder_prepend_item(XbelItem*, XbelItem*);
+
+void
+xbel_folder_remove_item(XbelItem*, XbelItem*);
+
+guint
+xbel_folder_get_n_items(XbelItem*);
+
+XbelItem*
+xbel_folder_get_nth_item(XbelItem*, guint);
+
+gboolean
+xbel_folder_is_empty(XbelItem*);
+
+gboolean
+xbel_folder_get_folded(XbelItem*);
+
+XbelItemKind
+xbel_item_get_kind(XbelItem*);
+
+XbelItem*
+xbel_item_get_parent(XbelItem*);
+
+G_CONST_RETURN gchar*
+xbel_item_get_title(XbelItem*);
+
+G_CONST_RETURN gchar*
+xbel_item_get_desc(XbelItem*);
+
+G_CONST_RETURN gchar*
+xbel_bookmark_get_href(XbelItem*);
+
+/*time_t
+xbel_bookmark_get_added(XbelItem*);
+
+time_t
+xbel_bookmark_get_modified(XbelItem*);
+
+time_t
+xbel_bookmark_get_visited(XbelItem*);*/
+
+gboolean
+xbel_item_is_bookmark(XbelItem*);
+
+gboolean
+xbel_item_is_separator(XbelItem*);
+
+gboolean
+xbel_item_is_folder(XbelItem*);
+
+void
+xbel_folder_set_folded(XbelItem*, gboolean);
+
+void
+xbel_item_set_title(XbelItem*, const gchar*);
+
+void
+xbel_item_set_desc(XbelItem*, const gchar*);
+
+void
+xbel_bookmark_set_href(XbelItem*, const gchar*);
+
+gboolean
+xbel_folder_from_data(XbelItem*, const gchar*, GError**);
+
+gboolean
+xbel_folder_from_file(XbelItem*, const gchar*, GError**);
+
+gboolean
+xbel_folder_from_data_dirs(XbelItem*, const gchar*, gchar**, GError**);
+
+gchar*
+xbel_folder_to_data(XbelItem*, gsize*, GError**);
+
+gboolean
+xbel_folder_to_file(XbelItem*, const gchar*, GError**);
+
+#endif /* !__XBEL_H__ */