Overview

Classes

  • _zp_captcha
  • _zp_HTML_cache
  • admin_approval
  • Album
  • AlbumBase
  • AlbumZip
  • AMFReader
  • AMFStream
  • AnyFile
  • AnyFile_Options
  • auto_backup
  • AVCSequenceParameterSetReader
  • bxslider
  • cacheManager
  • cachemanager_internal_deprecations
  • cacheManagerFeed
  • CI_jsmin
  • CI_load
  • cloneZenphoto
  • codeIgniter_kludge
  • colorbox
  • Combi
  • Comment
  • comment_form
  • contactformOptions
  • cookieConsent
  • crop_image
  • cycle
  • defaultCodeblocks
  • deprecated_functions
  • DownloadList
  • dynamic_locale
  • dynamicAlbum
  • elFinder
  • elFinder_options
  • elFinderConnector
  • elFinderEditor
  • elFinderEditorOnlineConvert
  • elFinderEditorZipArchive
  • elFinderEditorZohoOffice
  • elFinderLibGdBmp
  • elFinderPlugin
  • elFinderPluginAutoResize
  • elFinderPluginAutoRotate
  • elFinderPluginNormalizer
  • elFinderPluginSanitizer
  • elFinderPluginWatermark
  • elFinderSession
  • elFinderVolumeBox
  • elFinderVolumeDriver
  • elFinderVolumeDropbox
  • elFinderVolumeDropbox2
  • elFinderVolumeFlysystemGoogleDriveCache
  • elFinderVolumeFlysystemGoogleDriveNetmount
  • elFinderVolumeFTP
  • elFinderVolumeGoogleDrive
  • elFinderVolumeGroup
  • elFinderVolumeLocalFileSystem
  • elFinderVolumeMySQL
  • elFinderVolumeOneDrive
  • elFinderVolumeTrash
  • elFinderVolumeTrashMySQL
  • email_new_user
  • exampleMacros
  • external_auth
  • favorites
  • favoritesOptions
  • feed
  • fieldExtender
  • flag_thumbnail
  • Gallery
  • galleryArticles
  • getID3
  • getid3_aac
  • getid3_apetag
  • getid3_flv
  • getid3_handler
  • getid3_id3v1
  • getid3_id3v2
  • getid3_lib
  • getid3_lyrics3
  • getid3_mp3
  • getid3_mpeg
  • getid3_quicktime
  • getid3_swf
  • GoogleMap
  • Googlemaps
  • hitcounter
  • htmlmetatags
  • Image
  • internal_deprecations
  • ipBlocker
  • jPlayer
  • jplayer_options
  • jquery_rating
  • JSMin
  • lazyload
  • lib_GD_Options
  • lib_Imagick_Options
  • lib_NoGraphics
  • matomoStats
  • MediaObject
  • menu_manager
  • MergedRSS
  • MergedRSSOptions
  • mobile
  • Mobile_Detect
  • mobileTheme
  • multipleLayoutOptions
  • null_seo
  • OAuthConsumer
  • OAuthDataStore
  • OAuthRequest
  • OAuthServer
  • OAuthSignatureMethod
  • OAuthSignatureMethod_HMAC_SHA1
  • OAuthSignatureMethod_PLAINTEXT
  • OAuthSignatureMethod_RSA_SHA1
  • OAuthToken
  • OAuthUtil
  • openStreetMap
  • openStreetMapOptions
  • pagedThumbsNav
  • pagedthumbsOptions
  • PersistentObject
  • PHPMailer\PHPMailer\PHPMailer
  • PHPMailer\PHPMailer\POP3
  • PHPMailer\PHPMailer\SMTP
  • print_album_menu
  • pseudoPlayer
  • publishContent
  • quota_manager
  • reCaptcha
  • RecursiveCallbackFilterIterator
  • redirector
  • redirectorOptions
  • register_user
  • rewriteRules
  • rewriteTokens
  • RSS
  • rss_options
  • scriptlessSocialsharing
  • scriptlessSocialsharingOptions
  • search_statistics
  • SearchEngine
  • security_logger
  • securityHeaders
  • securityheadersOptions
  • seo_locale
  • Services_JSON
  • Services_JSON_Error
  • setup
  • setupMutex
  • setupRSS
  • show_not_loggedin
  • sitemap
  • sitemapOptions
  • static_html_cache
  • staticHTMLCacheOptions
  • tagsuggest
  • TextObject
  • TextObject_Options
  • ThemeObject
  • themeSwitcher
  • tinymce4Options
  • tinyURL
  • Transientimage
  • UploadHandler
  • user_expiry
  • user_groups
  • user_logout_options
  • userAddressFields
  • userDataExport
  • utf8
  • Video
  • VideoObject_Options
  • WEBdocs
  • WEBdocs_Options
  • xmpMetadata
  • Zenpage
  • Zenpage_internal_deprecations
  • ZenpageCategory
  • zenpagecms
  • ZenpageItems
  • ZenpageNews
  • ZenpagePage
  • ZenpageRoot
  • Zenphoto_Administrator
  • Zenphoto_Authority
  • zenphoto_org_news
  • zenphoto_seo
  • zenphotoDonate
  • ZipStream
  • zp_PHPMailer
  • zpFunctions
  • zpMutex
  • zpSimpleSpam
  • zpTrivialSpam

Interfaces

  • elFinderSessionInterface

Exceptions

  • elFinderAbortException
  • elFinderTriggerException
  • getid3_exception
  • JSMin_UnterminatedCommentException
  • JSMin_UnterminatedRegExpException
  • JSMin_UnterminatedStringException
  • OAuthExcept
  • PHPMailer\PHPMailer\Exception

Functions

  • access
  • accessAlbums
  • accessAllAlbums
  • accessImage
  • add_context
  • addalbumsToDatabase
  • addCategoriesToDatabase
  • addDateToTitlelink
  • addGeoCoord
  • addItem
  • addMissingDefaultRewriteTokens
  • addPagesToDatabase
  • addReconfigureNote
  • addSubalbumMenus
  • addWatermark
  • admin_album_list
  • admin_securityChecks
  • admin_showupdate
  • adminPageNav
  • adminToolbox
  • albumNumber
  • applyMacros
  • authorSelector
  • bind_textdomain_codeset
  • bindtextdomain
  • build_query
  • build_url
  • bulkActionRedirect
  • bulkTags
  • byteConvert
  • cacheImage
  • checkAccess
  • checkAlbumForImages
  • checkAlbumimagesort
  • checkAlbumParentid
  • checkAlbumPassword
  • checkChosenItemStatus
  • checkChosenMenuset
  • checked
  • checkFolder
  • checkForEmptyTitle
  • checkForGuest
  • checkForPage
  • checkForUpdate
  • checkHitcounterDisplay
  • checkIfChecked
  • checkIfLockedNews
  • checkIfLockedPage
  • checkIfNew
  • checkInstall
  • checkLayoutUseForImages
  • checkObjectsThumb
  • checkPageValidity
  • checkParentLayouts
  • checkPublishDates
  • checkRequiredField
  • checkSchedulePublishingNotes
  • checkSelectedAlbum
  • checkSignature
  • checkTitlelinkDuplicate
  • cleanAlbum
  • cleanHTML
  • clonedFrom
  • codeblocktabsJS
  • comment_form_addComment
  • comment_form_handle_comment
  • comment_form_PaginationJS
  • comment_form_postcomment
  • comment_form_print10Most
  • comment_form_visualEditor
  • commentFormUseCaptcha
  • commentReply
  • commentsAllowed
  • compressRow
  • consolidatedEditMessages
  • copyLayoutSelection
  • copyThemeDirectory
  • createMenuIfNotExists
  • createRelatedItemsResultArray
  • createTitlelink
  • cron_starter
  • curlRequest
  • currentRelativeURL
  • customOptions
  • dateDiff
  • datepickerJS
  • dateTimeConvert
  • db_affected_rows
  • db_close
  • db_collation
  • db_connect
  • db_count
  • db_create
  • db_create_table
  • db_error
  • db_fetch_assoc
  • db_fetch_row
  • db_free_result
  • db_getClientInfo
  • db_getServerInfo
  • db_getSQLmode
  • db_getVersion
  • db_insert_id
  • db_isMariaDB
  • db_LIKE_escape
  • db_list_fields
  • db_name
  • db_num_rows
  • db_permissions
  • db_quote
  • db_setSQLmode
  • db_show
  • db_software
  • db_table_update
  • db_truncate_table
  • debug404
  • debugLog
  • debugLogBacktrace
  • debuglogReconfigureNote
  • debugLogVar
  • decompressField
  • decompressRow
  • defaultCodeblocks_codebox
  • deleteArticle
  • deleteCategory
  • deleteItem
  • deleteLayoutSelection
  • deletePage
  • deleteThemeDirectory
  • dircopy
  • disableExtension
  • elFinder_admin_tabs
  • elFinder_tinymce
  • elFinderAutoloader
  • enableExtension
  • executeRSS
  • exitZP
  • exposeZenPhotoInformations
  • extendExecution
  • extensionEnabled
  • fetchComments
  • filesystemToInternal
  • fillbuffer
  • filterImageQuery
  • fix_path_redirect
  • formatList
  • fullText
  • galleryAlbumsPerPage
  • gallerystats_filesize_r
  • genAlbumList
  • generateAttributesFromArray
  • generateImageCacheFile
  • generateLanguageList
  • generateListFromArray
  • generateListFromFiles
  • generateRadiobuttonsFromArray
  • generateUnorderedListFromArray
  • get_AnyFile_suffixes
  • get_context
  • get_filterScript
  • get_instance
  • get_language_string
  • getAdminstratorsOptionsArray
  • getAdminThumb
  • getAdminThumbHTML
  • getAlbumArray
  • getAlbumBreadcrumb
  • getAlbumBreadcrumbAdmin
  • getAlbumCustomData
  • getAlbumData
  • getAlbumDate
  • getAlbumDesc
  • getAlbumFolder
  • getAlbumGeodata
  • getAlbumInherited
  • getAlbumLocation
  • getAlbumPage
  • getAlbumStatistic
  • getAlbumThumb
  • getAlbumTitle
  • getAlbumURL
  • getAllAccessibleAlbums
  • getAllAlbums
  • getAllDates
  • getAllowedTags
  • getAllSubAlbumIDs
  • getAllTagsCount
  • getAllTagsFromAlbum
  • getAllTagsFromAlbum_multi_unique
  • getAllTagsFromZenpage
  • getAllTagsUnique
  • getAllTranslations
  • getAnnotatedAlbumTitle
  • getAnnotatedImageTitle
  • getAnonymIP
  • getAuthor
  • getBare
  • getBareAlbumDesc
  • getBareAlbumTitle
  • getBareGalleryDesc
  • getBareGalleryTitle
  • getBareImageDesc
  • getBareImageTitle
  • getBareNewsTitle
  • getBarePageTitle
  • getCheckboxState
  • getCodeblock
  • getCommentAddress
  • getCommentAuthorEmail
  • getCommentAuthorLink
  • getCommentAuthorName
  • getCommentAuthorSite
  • getCommentBody
  • getCommentCount
  • getCommentDateTime
  • getCommentErrors
  • getCommentStored
  • getContactFormMacros
  • getContentShorten
  • getCookieInfoData
  • getCookieInfoHTML
  • getCookieInfoMacro
  • getCurrentMenuItem
  • getCurrentNewsArchive
  • getCurrentPage
  • getCurrentTheme
  • getCustomAlbumThumb
  • getCustomAlbumThumbMaxSpace
  • getCustomGalleryIndexPage
  • getCustomGalleryIndexURL
  • getCustomImageURL
  • getCustomPageURL
  • getCustomSizedImageMaxSpace
  • getCustomSizedImageThumbMaxSpace
  • getDataUsageNotice
  • getDefaultHeight
  • getDefaultRewriteTokens
  • getDefaultSizedImage
  • getDefaultWidth
  • getdownloadList
  • getDownloadURL
  • getE
  • getEnabledPlugins
  • getExpiryDatePost
  • getFavoritesURL
  • getField
  • getFullHeight
  • getFullImageFilesize
  • getFullImageURL
  • getFullWidth
  • getGalleryDesc
  • getGalleryIndexURL
  • getGalleryTitle
  • getGeoCoord
  • getHeadTitle
  • getHitcounter
  • getImageArgs
  • getImageCacheFilename
  • getImageCachePostfix
  • getImageCity
  • getImageCountry
  • getImageCustomData
  • getImageData
  • getImageDate
  • getImageDesc
  • getImageGeodata
  • getImageLocation
  • getImageMetaData
  • getImageParameters
  • getImageProcessorURI
  • getImageProcessorURIFromCacheName
  • getImageRotation
  • getImageState
  • getImageStatistic
  • getImageThumb
  • getImageTitle
  • getImageType
  • getImageURI
  • getImageURL
  • getItem
  • getItemByID
  • getItemTitleAndURL
  • getjPlayerSkinCSS
  • getjPlayerSkins
  • getLangAttributeLocale
  • getLanguageArray
  • getLanguageDisplayName
  • getLanguageFlag
  • getLanguageSubdomains
  • getLanguageText
  • getLatestComments
  • getLatestNews
  • getLatestZenpageComments
  • getLayout
  • getLayoutSelector
  • getLinkHTML
  • getLogTabs
  • getMacros
  • getMainSiteName
  • getMainSiteURL
  • getManagedAlbumList
  • getMaxSpaceContainer
  • getMenuFromLink
  • getMenuItemChilds
  • getMenuItems
  • getMenumanagerPredicessor
  • getMenumanagerSuccessor
  • getMenuSetSelector
  • getMenuVisibility
  • getMimeString
  • getNestedAlbumList
  • getNewsAdminOption
  • getNewsAdminOptionPath
  • getNewsArchivePath
  • getNewsArchiveURL
  • getNewsAuthor
  • getNewsCategories
  • getNewsCategoryCustomData
  • getNewsCategoryDesc
  • getNewsCategoryURL
  • getNewsContent
  • getNewsCustomData
  • getNewsDate
  • getNewsExtraContent
  • getNewsID
  • getNewsIndexURL
  • getNewsPagesStatistic
  • getNewsPathNav
  • getNewsReadMore
  • getNewsTitle
  • getNewsURL
  • getNextAlbum
  • getNextAlbumURL
  • getNextImageThumb
  • getNextImageURL
  • getNextNewsPageURL
  • getNextNewsURL
  • getNextPageURL
  • getNextPrevNews
  • getNotViewableAlbums
  • getNotViewableImages
  • getNumAlbums
  • getNumAllSubalbums
  • getNumImages
  • getNumNews
  • getNumPages
  • getOption
  • getOptionFromDB
  • getOptionList
  • getOwnerAuthor
  • getOwnerAuthorURL
  • getPageAuthor
  • getPageContent
  • getPageCustomData
  • getPageDate
  • getPageExtraContent
  • getPageID
  • getPageLastChangeDate
  • getPageNavList
  • getPageNumURL
  • getPageParentID
  • getPageRedirect
  • getPageSelector
  • getPageSortorder
  • getPageTitle
  • getPageTitleLink
  • getPageURL
  • getParentAlbums
  • getParentAlbumsAdmin
  • getParentBreadcrumb
  • getParentMenuItems
  • getPasswordProtectImage
  • getPHPFiles
  • getPictureOfTheDay
  • getPlugin
  • getPluginFiles
  • getPluginTabs
  • getPrevAlbum
  • getPrevAlbumURL
  • getPrevImageThumb
  • getPrevImageURL
  • getPrevNewsPageURL
  • getPrevNewsURL
  • getPrevPageURL
  • getProtectedImageURL
  • getRandomImages
  • getRandomImagesAlbum
  • getRating
  • getReconfigureNote
  • getRelatedItems
  • getRequestURI
  • getrow
  • getRSSLink
  • getRules
  • getSearchDate
  • getSearchURL
  • getSearchWords
  • getSelectedLayout
  • getSerializedArray
  • getSetClause
  • getSiteHomeURL
  • getSizeCustomImage
  • getSizeDefaultImage
  • getSizeDefaultThumb
  • getSizedImageURL
  • getSizeFullImage
  • getSortByOptions
  • getSortByStatusOptions
  • getStandardGalleryIndexURL
  • getSubtabs
  • getSuffix
  • getSystemLocales
  • getTagCountByAccess
  • getTagOrder
  • getTags
  • gettext
  • gettext_pl
  • gettext_th
  • getThemeFiles
  • getThemeOption
  • getTimezones
  • getTinyMCE4ConfigFiles
  • getTitle
  • getTotalHitcounter
  • getTotalImagesIn
  • getTotalNewsPages
  • getTotalPages
  • getUnprotectedImageURL
  • getUrAlbum
  • getUserIP
  • getUserLocale
  • getUserURL
  • getVersion
  • getWatermarkParam
  • getWatermarkPath
  • getWatermarks
  • getWhereClause
  • getXSRFToken
  • getZenpagePagesOptionsArray
  • getZenpageStatistic
  • handleSearchParms
  • hasDynamicAlbumSuffix
  • hasNextImage
  • hasNextNewsPage
  • hasNextPage
  • hasPrevImage
  • hasPrevNewsPage
  • hasPrevPage
  • hasPrimaryScripts
  • hl_attrval
  • hl_bal
  • hl_cmtcd
  • hl_ent
  • hl_prot
  • hl_regex
  • hl_spec
  • hl_tag
  • hl_tag2
  • hl_tidy
  • hl_version
  • html_decode
  • html_encode
  • html_encodeTagged
  • html_pathurlencode
  • htmLawed
  • httpsRedirect
  • httpUploadHandler
  • httpUploadHandler_admin_tabs
  • i18nSetLocale
  • ignoreSetupRunRequest
  • imageBlurGD
  • imagecreatefrombmp
  • imageDebug
  • imageError
  • imageNumber
  • imgSrcURI
  • in_context
  • installSignature
  • instrument
  • internalToFilesystem
  • inventMenuItem
  • iptc_make_tag
  • is_AdminEditPage
  • is_connected
  • is_News
  • is_NewsArchive
  • is_NewsArticle
  • is_NewsCategory
  • is_NewsPage
  • is_Pages
  • is_valid_email_zp
  • is_zip
  • isAlbumClass
  • isAlbumPage
  • isArchive
  • isCurrentitemParent
  • isHandledAlbum
  • isImageClass
  • isImagePage
  • isImagePhoto
  • isImageVideo
  • isIncompatibleExtension
  • isLandscape
  • isolate
  • isSetupProtected
  • isValidEmail
  • isValidURL
  • jQueryUpload_head
  • jQueryUpload_headers
  • jQueryUploadHandler
  • jQueryUploadHandler_admin_tabs
  • js_encode
  • json_decode
  • json_encode
  • kses
  • kses_array_lc
  • kses_attr
  • kses_bad_protocol
  • kses_bad_protocol_once
  • kses_bad_protocol_once2
  • kses_check_attr_val
  • kses_decode_entities
  • kses_hair
  • kses_hook
  • kses_html_error
  • kses_js_entities
  • kses_no_null
  • kses_normalize_entities
  • kses_normalize_entities2
  • kses_split
  • kses_split2
  • kses_stripslashes
  • kses_version
  • ksesProcess
  • layoutSelector
  • layoutSelector_album
  • listDBUses
  • listDirectoryFiles
  • listUses
  • load_zenpage_news
  • load_zenpage_pages
  • loadLocalOptions
  • log_message
  • lookupSortKey
  • macro_admin_tabs
  • macroList_show
  • makeAlbumCurrent
  • makeImageCurrent
  • makeSpecialImageName
  • markRelease_button
  • mb_strlen
  • mb_strpos
  • mb_strrpos
  • mb_strtolower
  • mb_strtoupper
  • mb_substr
  • mb_substr_count
  • menu_admin_toolbox_global
  • menu_tabs
  • minDiff
  • mkdir_recursive
  • myts_date
  • newAlbum
  • newImage
  • next_album
  • next_comment
  • next_image
  • next_news
  • next_page
  • ngettext
  • ngettext_pl
  • ngettext_th
  • omsAdditions
  • parse_query
  • parse_size
  • parseAllowedTags
  • parseHttpAcceptLanguage
  • pathurlencode
  • pluginDebug
  • populateManagedObjectsList
  • postAlbumSort
  • postIndexDecode
  • postIndexEncode
  • prefix
  • prepareAlbumPage
  • prepareCustomPage
  • prepareImagePage
  • prepareIndexPage
  • print404status
  • print_language_string_list
  • printAddToFavorites
  • printAdminFooter
  • printAdminHeader
  • printAdminRightsTable
  • printAdminThumb
  • printAlbumBreadcrumb
  • printAlbumButtons
  • printAlbumCustomData
  • printAlbumData
  • printAlbumDate
  • printAlbumDesc
  • printAlbumEditForm
  • printAlbumEditRow
  • printAlbumLegend
  • printAlbumLocation
  • printAlbumMenu
  • printAlbumMenuJump
  • printAlbumMenuList
  • printAlbumMenuListAlbum
  • printAlbumsSelector
  • printAlbumStatistic
  • printAlbumStatisticItem
  • printAlbumThumbImage
  • printAlbumTitle
  • printAlbumURL
  • printAllDates
  • printAllNestedList
  • printAllNewsCategories
  • printAllTags
  • printAllTagsAs
  • printAllTagsFromAlbum
  • printAllTagsFromZenpage
  • printAnnotatedAlbumTitle
  • printAnnotatedImageTitle
  • printArticleCategories
  • printArticleDatesDropdown
  • printArticlesPerPageDropdown
  • printAuthorDropdown
  • printBareAlbumDesc
  • printBareAlbumTitle
  • printBareGalleryDesc
  • printBareGalleryTitle
  • printBareImageDesc
  • printBareImageTitle
  • printBareNewsTitle
  • printBarePageTitle
  • printBarGraph
  • printBulkActions
  • printCategoriesStatistic
  • printCategoryCheckboxListEntry
  • printCategoryDropdown
  • printCategoryListSortableTable
  • printCategorySelection
  • printCodeblock
  • printCodeblockEdit
  • printCommentAuthorLink
  • printCommentErrors
  • printCommentForm
  • printContactForm
  • printContactFormMacro
  • printCookieInfo
  • printCopyrightNotice
  • printCurrentNewsArchive
  • printCurrentNewsCategory
  • printCustomAlbumThumbImage
  • printCustomAlbumThumbMaxSpace
  • printCustomMenu
  • printCustomPageSelector
  • printCustomPageURL
  • printCustomSizedImage
  • printCustomSizedImageMaxSpace
  • printCustomSizedImageThumbMaxSpace
  • printDataUsageNotice
  • printDefaultSizedImage
  • printDownloadAlbumZipURL
  • printdownloadList
  • printDownloadURL
  • printEditCommentLink
  • printEditDropdown
  • printExpired
  • printFavoritesURL
  • printFullAlbumsList
  • printFullImageDownloadURL
  • printGalleryDesc
  • printGalleryIndexURL
  • printGalleryTitle
  • printGoogleMap
  • printHeadTitle
  • printHomeLink
  • printImageCustomData
  • printImageData
  • printImageDate
  • printImageDesc
  • printImageMetadata
  • printImageslist
  • printImageStatistic
  • printImageThumb
  • printImageTitle
  • printImageURL
  • printItemEditLink
  • printItemsList
  • printItemsListTable
  • printItemStatusDropdown
  • printjPlayerPlaylist
  • printLangAttribute
  • printLanguageSelector
  • printLastChangeInfo
  • printLatestAlbums
  • printLatestComments
  • printLatestImages
  • printLatestImagesByDate
  • printLatestImagesByMtime
  • printLatestNews
  • printLatestUpdatedAlbums
  • printLinkHTML
  • printLogoAndLinks
  • printManagedObjects
  • printMenuemanagerPageList
  • printMenuemanagerPageListWithNav
  • printMenumanagerBreadcrumb
  • printMenumanagerNextLink
  • printMenumanagerPrevLink
  • printMostPopularItems
  • printMostRatedAlbums
  • printMostRatedImages
  • printMostRatedItems
  • printNestedAlbumsList
  • printNestedItemsList
  • printNestedMenu
  • printNews
  • printNewsArchive
  • printNewsArticlesList
  • printNewsAuthor
  • printNewsCategories
  • printNewsCategoryCustomData
  • printNewsCategoryDesc
  • printNewsCategoryURL
  • printNewsContent
  • printNewsCustomData
  • printNewsDate
  • printNewsExtraContent
  • printNewsIndexURL
  • printNewsPageList
  • printNewsPageListWithNav
  • printNewsStatistic
  • printNewsTitle
  • printNewsURL
  • printNextNewsLink
  • printNextNewsPageLink
  • printNextPageURL
  • printOpenStreetMap
  • printOwnerAuthor
  • printOwnerAuthorURL
  • printPageArticleTags
  • printPageAuthor
  • printPageContent
  • printPageCustomData
  • printPageDate
  • printPagedThumbsNav
  • printPageExtraContent
  • printPageID
  • printPageLastChangeDate
  • printPageList
  • printPageListWithNav
  • printPageMenu
  • printPageNav
  • printPageSelector
  • printPagesListTable
  • printPagesStatistic
  • printPageTitle
  • printPageTitleLink
  • printPageURL
  • printParentBreadcrumb
  • printPasswordForm
  • printPopularAlbums
  • printPopularImages
  • printPrevNewsLink
  • printPrevNewsPageLink
  • printPrevPageURL
  • printPrivacyPageLink
  • printPublished
  • printPublishIconLink
  • printPublishIconLinkGallery
  • printRandomImages
  • printRating
  • printReconfigureError
  • printReconfigureNote
  • printRegisterURL
  • printRegistrationForm
  • printRelatedItems
  • printRSSHeaderLink
  • printRSSLink
  • printScheduledPublishingNotes
  • printSearchBreadcrumb
  • printSearchForm
  • printSelectorWithCustomField
  • printSiteHomeURL
  • printSizedImageURL
  • printSlideShow
  • printSlideShowLink
  • printSortableHead
  • printSortOrderDropdown
  • printSubLevelAlbums
  • printSubPagesExcerpts
  • printSubtabs
  • printTabs
  • printTags
  • printThumbNav
  • printTinyPageNav
  • printTinyZenpageCategorySelector
  • printTopRatedAlbums
  • printTopRatedImages
  • printTopRatedItems
  • printUnpublishedDropdown
  • printUserLogin_out
  • printUserSelector
  • printUserURL
  • printVersion
  • printZenJavascripts
  • printZenpageIconLegend
  • printZenpageItems
  • printZenpageItemsBreadcrumb
  • printZenpageNewsCategorySelector
  • printZenpagePageSelector
  • printZenpagePagesSelector
  • printZenpageStatistic
  • printZenphotoLink
  • process_language_string_save
  • processAlbumBulkActions
  • processAlbumEdit
  • processCodeblockSave
  • processCommentBulkActions
  • processCredentials
  • processCustomOptionSave
  • processEditSelection
  • processExtensionVariable
  • processImageBulkActions
  • processImageEdit
  • processManagedObjects
  • processMenuBulkActions
  • processOrder
  • processRights
  • processTags
  • processZenpageBulkActions
  • propSizes
  • protectSetupFiles
  • publishItem
  • purgeOption
  • purgeThemeOption
  • query
  • query_full_array
  • query_single_row
  • rc4
  • read_exif_data_protected
  • readTags
  • reconfigureAction
  • reconfigureCSS
  • recordMissing
  • redirectionHandler
  • redirectURL
  • rem_context
  • removeDir
  • removeParentAlbumNames
  • removeTrailingSlash
  • renameOption
  • replaceOption
  • replaceThemeOption
  • restore_context
  • reveal
  • rewrite_get_album_image
  • rewrite_path
  • rewriteHandler
  • RSS_Channel
  • RSS_Retrieve
  • RSS_Tags
  • rulesList
  • safe_fnmatch
  • safe_glob
  • sanitize
  • sanitize_numeric
  • sanitize_path
  • sanitize_script
  • sanitize_string
  • sanitizeRedirect
  • save_context
  • saveLayoutSelection
  • saveZenphotoLayoutSelection
  • search_quote
  • secureServer
  • seo_cleanup_button
  • seoFriendly
  • seoFriendlyJS
  • set_context
  • setAlbumSubtabs
  • setexifvars
  • setMainDomain
  • setOption
  • setOptionDefault
  • setSiteState
  • setThemeColumns
  • setThemeOption
  • setThemeOptionDefault
  • setTinyZenpageLocale
  • setupCurrentLocale
  • setupDomain
  • setupTheme
  • shortenContent
  • shortentitle
  • showOrNotShowField
  • showZenphotoOptions
  • shuffle_assoc
  • signatureChange
  • site_upgrade_button
  • site_upgrade_status
  • skipScheduledPublishing
  • sortArray
  • sortByKey
  • sortByMultilingual
  • sortMultiArray
  • standardScripts
  • standardThemeOptions
  • stickyNews
  • storeConfig
  • storeTags
  • stripSuffix
  • submenuOf
  • switchLog
  • tagSelector
  • tagSuggestJS
  • tagSuggestJS_admin
  • tagSuggestJS_frontend
  • tagURLs
  • textdomain
  • themeIsEditable
  • themeSetup
  • tidyHTML
  • timezoneDiff
  • tinymce4ConfigJS
  • truncate_string
  • unprotectSetupFiles
  • unpublishedZenpageItemCheck
  • unpublishedZenphotoItemCheck
  • unpublishSubalbums
  • unQuote
  • unTagURLs
  • unzip
  • updateArticle
  • updateCacheName
  • updateCategory
  • updateConfigItem
  • updateImageProcessorLink
  • updateItemSortorder
  • updateItemsSortorder
  • updateMenuItem
  • updatePage
  • upload_extra
  • upload_form
  • upload_head
  • user_mailing_list_button
  • validateLocale
  • writeHeader
  • XSRFdefender
  • XSRFToken
  • zenpageAlbumImage
  • zenpageBulkActionMessage
  • zenpageJSCSS
  • zenpagePublish
  • zenphoto_PHPMailer
  • zenphoto_sendmail
  • zenPhotoTheme
  • zp_apply_filter
  • zp_clearCookie
  • zp_colorAllocate
  • zp_cookieEncode
  • zp_copyCanvas
  • zp_createImage
  • zp_drawRectangle
  • zp_error
  • zp_filter_slot
  • zp_filter_unique_id
  • zp_getCookie
  • zp_getFonts
  • zp_graphicsLibInfo
  • zp_handle_password
  • zp_handle_password_single
  • zp_has_filter
  • zp_image_types
  • zp_imageCanRotate
  • zp_imageColorTransparent
  • zp_imageDims
  • zp_imageFill
  • zp_imageFontHeight
  • zp_imageFontWidth
  • zp_imageFromString
  • zp_imageGet
  • zp_imageGray
  • zp_imageHeight
  • zp_imageIPTC
  • zp_imageKill
  • zp_imageLoadFont
  • zp_imageMerge
  • zp_imageOutput
  • zp_imageResizeAlpha
  • zp_imageResizeTransparent
  • zp_imageUnsharpMask
  • zp_imageWidth
  • zp_load_album
  • zp_load_gallery
  • zp_load_image
  • zp_load_page
  • zp_load_request
  • zp_load_search
  • zp_loggedin
  • zp_mail
  • zp_register_filter
  • zp_remove_filter
  • zp_resampleImage
  • zp_rotateImage
  • zp_session_destroy
  • zp_session_start
  • zp_setCookie
  • zp_writeString
  • zpErrorHandler
  • zpFormattedDate
  • zpRewriteURL
  • Overview
  • Class
  • Tree
  • Package
  • Deprecated
   1:    2:    3:    4:    5:    6:    7:    8:    9:   10:   11:   12:   13:   14:   15:   16:   17:   18:   19:   20:   21:   22:   23:   24:   25:   26:   27:   28:   29:   30:   31:   32:   33:   34:   35:   36:   37:   38:   39:   40:   41:   42:   43:   44:   45:   46:   47:   48:   49:   50:   51:   52:   53:   54:   55:   56:   57:   58:   59:   60:   61:   62:   63:   64:   65:   66:   67:   68:   69:   70:   71:   72:   73:   74:   75:   76:   77:   78:   79:   80:   81:   82:   83:   84:   85:   86:   87:   88:   89:   90:   91:   92:   93:   94:   95:   96:   97:   98:   99:  100:  101:  102:  103:  104:  105:  106:  107:  108:  109:  110:  111:  112:  113:  114:  115:  116:  117:  118:  119:  120:  121:  122:  123:  124:  125:  126:  127:  128:  129:  130:  131:  132:  133:  134:  135:  136:  137:  138:  139:  140:  141:  142:  143:  144:  145:  146:  147:  148:  149:  150:  151:  152:  153:  154:  155:  156:  157:  158:  159:  160:  161:  162:  163:  164:  165:  166:  167:  168:  169:  170:  171:  172:  173:  174:  175:  176:  177:  178:  179:  180:  181:  182:  183:  184:  185:  186:  187:  188:  189:  190:  191:  192:  193:  194:  195:  196:  197:  198:  199:  200:  201:  202:  203:  204:  205:  206:  207:  208:  209:  210:  211:  212:  213:  214:  215:  216:  217:  218:  219:  220:  221:  222:  223:  224:  225:  226:  227:  228:  229:  230:  231:  232:  233:  234:  235:  236:  237:  238:  239:  240:  241:  242:  243:  244:  245:  246:  247:  248:  249:  250:  251:  252:  253:  254:  255:  256:  257:  258:  259:  260:  261:  262:  263:  264:  265:  266:  267:  268:  269:  270:  271:  272:  273:  274:  275:  276:  277:  278:  279:  280:  281:  282:  283:  284:  285:  286:  287:  288:  289:  290:  291:  292:  293:  294:  295:  296:  297:  298:  299:  300:  301:  302:  303:  304:  305:  306:  307:  308:  309:  310:  311:  312:  313:  314:  315:  316:  317:  318:  319:  320:  321:  322:  323:  324:  325:  326:  327:  328:  329:  330:  331:  332:  333:  334:  335:  336:  337:  338:  339:  340:  341:  342:  343:  344:  345:  346:  347:  348:  349:  350:  351:  352:  353:  354:  355:  356:  357:  358:  359:  360:  361:  362:  363:  364:  365:  366:  367:  368:  369:  370:  371:  372:  373:  374:  375:  376:  377:  378:  379:  380:  381:  382:  383:  384:  385:  386:  387:  388:  389:  390:  391:  392:  393:  394:  395:  396:  397:  398:  399:  400:  401:  402:  403:  404:  405:  406:  407:  408:  409:  410:  411:  412:  413:  414:  415:  416:  417:  418:  419:  420:  421:  422:  423:  424:  425:  426:  427:  428:  429:  430:  431:  432:  433:  434:  435:  436:  437:  438:  439:  440:  441:  442:  443:  444:  445:  446:  447:  448:  449:  450:  451:  452:  453:  454:  455:  456:  457:  458:  459:  460:  461:  462:  463:  464:  465:  466:  467:  468:  469:  470:  471:  472:  473:  474:  475:  476:  477:  478:  479:  480:  481:  482:  483:  484:  485:  486:  487:  488:  489:  490:  491:  492:  493:  494:  495:  496:  497:  498:  499:  500:  501:  502:  503:  504:  505:  506:  507:  508:  509:  510:  511:  512:  513:  514:  515:  516:  517:  518:  519:  520:  521:  522:  523:  524:  525:  526:  527:  528:  529:  530:  531:  532:  533:  534:  535:  536:  537:  538:  539:  540:  541:  542:  543:  544:  545:  546:  547:  548:  549:  550:  551:  552:  553:  554:  555:  556:  557:  558:  559:  560:  561:  562:  563:  564:  565:  566:  567:  568:  569:  570:  571:  572:  573:  574:  575:  576:  577:  578:  579:  580:  581:  582:  583:  584:  585:  586:  587:  588:  589:  590:  591:  592:  593:  594:  595:  596:  597:  598:  599:  600:  601:  602:  603:  604:  605:  606:  607:  608:  609:  610:  611:  612:  613:  614:  615:  616:  617:  618:  619:  620:  621:  622:  623:  624:  625:  626:  627:  628:  629:  630:  631:  632:  633:  634:  635:  636:  637:  638:  639:  640:  641:  642:  643:  644:  645:  646:  647:  648:  649:  650:  651:  652:  653:  654:  655:  656:  657:  658:  659:  660:  661:  662:  663:  664:  665:  666:  667:  668:  669:  670:  671:  672:  673:  674:  675:  676:  677:  678:  679:  680:  681:  682:  683:  684:  685:  686:  687:  688:  689:  690:  691:  692:  693:  694:  695:  696:  697:  698:  699:  700:  701:  702:  703:  704:  705:  706:  707:  708:  709:  710:  711:  712:  713:  714:  715:  716:  717:  718:  719:  720:  721:  722:  723:  724:  725:  726:  727:  728:  729:  730:  731:  732:  733:  734:  735:  736:  737:  738:  739:  740:  741:  742:  743:  744:  745:  746:  747:  748:  749:  750:  751:  752:  753:  754:  755:  756:  757:  758:  759:  760:  761:  762:  763:  764:  765:  766:  767:  768:  769:  770:  771:  772:  773:  774:  775:  776:  777:  778:  779:  780:  781:  782:  783:  784:  785:  786:  787:  788:  789:  790:  791:  792:  793:  794:  795:  796:  797:  798:  799:  800:  801:  802:  803:  804:  805:  806:  807:  808:  809:  810:  811:  812:  813:  814:  815:  816:  817:  818:  819:  820:  821:  822:  823:  824:  825:  826:  827:  828:  829:  830:  831:  832:  833:  834:  835:  836:  837:  838:  839:  840:  841:  842:  843:  844:  845:  846:  847:  848:  849:  850:  851:  852:  853:  854:  855:  856:  857:  858:  859:  860:  861:  862:  863:  864:  865:  866:  867:  868:  869:  870:  871:  872:  873:  874:  875:  876:  877:  878:  879:  880:  881:  882:  883:  884:  885:  886:  887:  888:  889:  890:  891:  892:  893:  894:  895:  896:  897:  898:  899:  900:  901:  902:  903:  904:  905:  906:  907:  908:  909:  910:  911:  912:  913:  914:  915:  916:  917:  918:  919:  920:  921:  922:  923:  924:  925:  926:  927:  928:  929:  930:  931:  932:  933:  934:  935:  936:  937:  938:  939:  940:  941:  942:  943:  944:  945:  946:  947:  948:  949:  950:  951:  952:  953:  954:  955:  956:  957:  958:  959:  960:  961:  962:  963:  964:  965:  966:  967:  968:  969:  970:  971:  972:  973:  974:  975:  976:  977:  978:  979:  980:  981:  982:  983:  984:  985:  986:  987:  988:  989:  990:  991:  992:  993:  994:  995:  996:  997:  998:  999: 1000: 1001: 1002: 1003: 1004: 1005: 1006: 1007: 1008: 1009: 1010: 1011: 1012: 1013: 1014: 1015: 1016: 1017: 1018: 1019: 1020: 1021: 1022: 1023: 1024: 1025: 1026: 1027: 1028: 1029: 1030: 1031: 1032: 1033: 1034: 1035: 1036: 1037: 1038: 1039: 1040: 1041: 1042: 1043: 1044: 1045: 1046: 1047: 1048: 1049: 1050: 1051: 1052: 1053: 1054: 1055: 1056: 1057: 1058: 1059: 1060: 1061: 1062: 1063: 1064: 1065: 1066: 1067: 1068: 1069: 1070: 1071: 1072: 1073: 1074: 1075: 1076: 1077: 1078: 1079: 1080: 1081: 1082: 1083: 1084: 1085: 1086: 1087: 1088: 1089: 1090: 1091: 1092: 1093: 1094: 1095: 1096: 1097: 1098: 1099: 1100: 1101: 1102: 1103: 1104: 1105: 1106: 1107: 1108: 1109: 1110: 1111: 1112: 1113: 1114: 1115: 1116: 1117: 1118: 1119: 1120: 1121: 1122: 1123: 1124: 1125: 1126: 1127: 1128: 1129: 1130: 1131: 1132: 1133: 1134: 1135: 1136: 1137: 1138: 1139: 1140: 1141: 1142: 1143: 1144: 1145: 1146: 1147: 1148: 1149: 1150: 1151: 1152: 1153: 1154: 1155: 1156: 1157: 1158: 1159: 1160: 1161: 1162: 1163: 1164: 1165: 1166: 1167: 1168: 1169: 1170: 1171: 1172: 1173: 1174: 1175: 1176: 1177: 1178: 1179: 1180: 1181: 1182: 1183: 1184: 1185: 1186: 1187: 1188: 1189: 1190: 1191: 1192: 1193: 1194: 1195: 1196: 1197: 1198: 1199: 1200: 1201: 1202: 1203: 1204: 1205: 1206: 1207: 1208: 1209: 1210: 1211: 1212: 1213: 1214: 1215: 1216: 1217: 1218: 1219: 1220: 1221: 1222: 1223: 1224: 1225: 1226: 1227: 1228: 1229: 1230: 1231: 1232: 1233: 1234: 1235: 1236: 1237: 1238: 1239: 1240: 1241: 1242: 1243: 1244: 1245: 1246: 1247: 1248: 1249: 1250: 1251: 1252: 1253: 1254: 1255: 1256: 1257: 1258: 1259: 1260: 1261: 1262: 1263: 1264: 1265: 1266: 1267: 1268: 1269: 1270: 1271: 1272: 1273: 1274: 1275: 1276: 1277: 1278: 1279: 1280: 1281: 1282: 1283: 1284: 1285: 1286: 1287: 1288: 1289: 1290: 1291: 1292: 1293: 1294: 1295: 1296: 1297: 1298: 1299: 1300: 1301: 1302: 1303: 1304: 1305: 1306: 1307: 1308: 1309: 1310: 1311: 1312: 1313: 1314: 1315: 1316: 1317: 1318: 1319: 1320: 1321: 1322: 1323: 1324: 1325: 1326: 1327: 1328: 1329: 1330: 1331: 1332: 1333: 1334: 1335: 1336: 1337: 1338: 1339: 1340: 1341: 1342: 1343: 1344: 1345: 1346: 1347: 1348: 1349: 1350: 1351: 1352: 1353: 1354: 1355: 1356: 1357: 1358: 1359: 1360: 1361: 1362: 1363: 1364: 1365: 1366: 1367: 1368: 1369: 1370: 1371: 1372: 1373: 1374: 1375: 1376: 1377: 1378: 1379: 1380: 1381: 1382: 1383: 1384: 1385: 1386: 1387: 1388: 1389: 1390: 1391: 1392: 1393: 1394: 1395: 1396: 1397: 1398: 1399: 1400: 1401: 1402: 1403: 1404: 1405: 1406: 1407: 1408: 1409: 1410: 1411: 1412: 1413: 1414: 1415: 1416: 1417: 1418: 1419: 1420: 1421: 1422: 1423: 1424: 1425: 1426: 1427: 1428: 1429: 1430: 1431: 1432: 1433: 1434: 1435: 1436: 1437: 1438: 1439: 1440: 1441: 1442: 1443: 1444: 1445: 1446: 1447: 1448: 1449: 1450: 1451: 1452: 1453: 1454: 1455: 1456: 1457: 1458: 1459: 1460: 1461: 1462: 1463: 1464: 1465: 1466: 1467: 1468: 1469: 1470: 1471: 1472: 1473: 1474: 1475: 1476: 1477: 1478: 1479: 1480: 1481: 1482: 1483: 1484: 1485: 1486: 1487: 1488: 1489: 1490: 1491: 1492: 1493: 1494: 1495: 1496: 1497: 1498: 1499: 1500: 1501: 1502: 1503: 1504: 1505: 1506: 1507: 1508: 1509: 1510: 1511: 1512: 1513: 1514: 1515: 1516: 1517: 1518: 1519: 1520: 1521: 1522: 1523: 1524: 1525: 1526: 1527: 1528: 1529: 1530: 1531: 1532: 1533: 1534: 1535: 1536: 1537: 1538: 1539: 1540: 1541: 1542: 1543: 1544: 1545: 1546: 1547: 1548: 1549: 1550: 1551: 1552: 1553: 1554: 1555: 1556: 1557: 1558: 1559: 1560: 1561: 1562: 1563: 1564: 1565: 1566: 1567: 1568: 1569: 1570: 1571: 1572: 1573: 1574: 1575: 1576: 1577: 1578: 1579: 1580: 1581: 1582: 1583: 1584: 1585: 1586: 1587: 1588: 1589: 1590: 1591: 1592: 1593: 1594: 1595: 1596: 1597: 1598: 1599: 1600: 1601: 1602: 1603: 1604: 1605: 1606: 1607: 1608: 1609: 1610: 1611: 1612: 1613: 1614: 1615: 1616: 1617: 1618: 1619: 1620: 1621: 1622: 1623: 1624: 1625: 1626: 1627: 1628: 1629: 1630: 1631: 1632: 1633: 1634: 1635: 1636: 1637: 1638: 1639: 1640: 1641: 1642: 1643: 1644: 1645: 1646: 1647: 1648: 1649: 1650: 1651: 1652: 1653: 1654: 1655: 1656: 1657: 1658: 1659: 1660: 1661: 1662: 1663: 1664: 1665: 1666: 1667: 1668: 1669: 1670: 1671: 1672: 1673: 1674: 1675: 1676: 1677: 1678: 1679: 1680: 1681: 1682: 1683: 1684: 1685: 1686: 1687: 1688: 1689: 1690: 1691: 1692: 1693: 1694: 1695: 1696: 1697: 1698: 1699: 1700: 1701: 1702: 1703: 1704: 1705: 1706: 1707: 1708: 1709: 1710: 1711: 1712: 1713: 1714: 1715: 1716: 1717: 1718: 1719: 1720: 1721: 1722: 1723: 1724: 1725: 1726: 1727: 1728: 1729: 1730: 1731: 1732: 1733: 1734: 1735: 1736: 1737: 1738: 1739: 1740: 1741: 1742: 1743: 1744: 1745: 1746: 1747: 1748: 1749: 1750: 1751: 1752: 1753: 1754: 1755: 1756: 1757: 1758: 1759: 1760: 1761: 1762: 1763: 1764: 1765: 1766: 1767: 1768: 1769: 1770: 1771: 1772: 1773: 1774: 1775: 1776: 1777: 1778: 1779: 1780: 1781: 1782: 1783: 1784: 1785: 1786: 1787: 1788: 1789: 1790: 1791: 1792: 1793: 1794: 1795: 1796: 1797: 1798: 1799: 1800: 1801: 1802: 1803: 1804: 1805: 1806: 1807: 1808: 1809: 1810: 1811: 1812: 1813: 1814: 1815: 1816: 1817: 1818: 1819: 1820: 1821: 1822: 1823: 1824: 1825: 1826: 1827: 1828: 1829: 1830: 1831: 1832: 1833: 1834: 1835: 1836: 1837: 1838: 1839: 1840: 1841: 1842: 1843: 1844: 1845: 1846: 1847: 1848: 1849: 1850: 1851: 1852: 1853: 1854: 1855: 1856: 1857: 1858: 1859: 1860: 1861: 1862: 1863: 1864: 1865: 1866: 1867: 1868: 1869: 1870: 1871: 1872: 1873: 1874: 1875: 1876: 1877: 1878: 1879: 1880: 1881: 1882: 1883: 1884: 1885: 1886: 1887: 1888: 1889: 1890: 1891: 1892: 1893: 1894: 1895: 1896: 1897: 1898: 1899: 1900: 1901: 1902: 1903: 1904: 1905: 1906: 1907: 1908: 1909: 1910: 1911: 1912: 1913: 1914: 1915: 1916: 1917: 1918: 1919: 1920: 1921: 1922: 1923: 1924: 1925: 1926: 1927: 1928: 1929: 1930: 1931: 1932: 1933: 1934: 1935: 1936: 1937: 1938: 1939: 1940: 1941: 1942: 1943: 1944: 1945: 1946: 1947: 1948: 1949: 1950: 1951: 1952: 1953: 1954: 1955: 1956: 1957: 1958: 1959: 1960: 1961: 1962: 1963: 1964: 1965: 1966: 1967: 1968: 1969: 1970: 1971: 1972: 1973: 1974: 1975: 1976: 1977: 1978: 1979: 1980: 1981: 1982: 1983: 1984: 1985: 1986: 1987: 1988: 1989: 1990: 1991: 1992: 1993: 1994: 1995: 1996: 1997: 1998: 1999: 2000: 2001: 2002: 2003: 2004: 2005: 2006: 2007: 2008: 2009: 2010: 2011: 2012: 2013: 2014: 2015: 2016: 2017: 2018: 2019: 2020: 2021: 2022: 2023: 2024: 2025: 2026: 2027: 2028: 2029: 2030: 2031: 2032: 2033: 2034: 2035: 2036: 2037: 2038: 2039: 2040: 2041: 2042: 2043: 2044: 2045: 2046: 2047: 2048: 2049: 2050: 2051: 2052: 2053: 2054: 2055: 2056: 2057: 2058: 2059: 2060: 2061: 2062: 2063: 2064: 2065: 2066: 2067: 2068: 2069: 2070: 2071: 2072: 2073: 2074: 2075: 2076: 2077: 2078: 2079: 2080: 2081: 2082: 2083: 2084: 2085: 2086: 2087: 2088: 2089: 2090: 2091: 2092: 2093: 2094: 2095: 2096: 2097: 2098: 2099: 2100: 2101: 2102: 2103: 2104: 2105: 2106: 2107: 2108: 2109: 2110: 2111: 2112: 2113: 2114: 2115: 2116: 2117: 2118: 2119: 2120: 2121: 2122: 2123: 2124: 2125: 2126: 2127: 2128: 2129: 2130: 2131: 2132: 2133: 2134: 2135: 2136: 2137: 2138: 2139: 2140: 2141: 2142: 2143: 2144: 2145: 2146: 2147: 2148: 2149: 2150: 2151: 2152: 2153: 2154: 2155: 2156: 2157: 2158: 2159: 2160: 2161: 2162: 2163: 2164: 2165: 2166: 2167: 2168: 2169: 2170: 2171: 2172: 2173: 2174: 2175: 2176: 2177: 2178: 2179: 2180: 2181: 2182: 2183: 2184: 2185: 2186: 2187: 2188: 2189: 2190: 2191: 2192: 2193: 2194: 2195: 2196: 2197: 2198: 2199: 2200: 2201: 2202: 2203: 2204: 2205: 2206: 2207: 2208: 2209: 2210: 2211: 2212: 2213: 2214: 2215: 2216: 2217: 2218: 2219: 2220: 2221: 2222: 2223: 2224: 2225: 2226: 2227: 2228: 2229: 2230: 2231: 2232: 2233: 2234: 2235: 2236: 2237: 2238: 2239: 2240: 2241: 2242: 2243: 2244: 2245: 2246: 2247: 2248: 2249: 2250: 2251: 2252: 2253: 2254: 2255: 2256: 2257: 2258: 2259: 2260: 2261: 2262: 2263: 2264: 2265: 2266: 2267: 2268: 2269: 2270: 2271: 2272: 2273: 2274: 2275: 2276: 2277: 2278: 2279: 2280: 2281: 2282: 2283: 2284: 2285: 2286: 2287: 2288: 2289: 2290: 2291: 2292: 2293: 2294: 2295: 2296: 2297: 2298: 2299: 2300: 2301: 2302: 2303: 2304: 2305: 2306: 2307: 2308: 2309: 2310: 2311: 2312: 2313: 2314: 2315: 2316: 2317: 2318: 2319: 2320: 2321: 2322: 2323: 2324: 2325: 2326: 2327: 2328: 2329: 2330: 2331: 2332: 2333: 2334: 2335: 2336: 2337: 2338: 2339: 2340: 2341: 2342: 2343: 2344: 2345: 2346: 2347: 2348: 2349: 2350: 2351: 2352: 2353: 2354: 2355: 2356: 2357: 2358: 2359: 2360: 2361: 2362: 2363: 2364: 2365: 2366: 2367: 2368: 2369: 2370: 2371: 2372: 2373: 2374: 2375: 2376: 2377: 2378: 2379: 2380: 2381: 2382: 2383: 2384: 2385: 2386: 2387: 2388: 2389: 2390: 2391: 2392: 2393: 2394: 2395: 2396: 2397: 2398: 2399: 2400: 2401: 2402: 2403: 2404: 2405: 2406: 2407: 2408: 2409: 2410: 2411: 2412: 2413: 2414: 2415: 2416: 2417: 2418: 2419: 2420: 2421: 2422: 2423: 2424: 2425: 2426: 2427: 2428: 2429: 2430: 2431: 2432: 2433: 2434: 2435: 2436: 2437: 2438: 2439: 2440: 2441: 2442: 2443: 2444: 2445: 2446: 2447: 2448: 2449: 2450: 2451: 2452: 2453: 2454: 2455: 2456: 2457: 2458: 2459: 2460: 2461: 2462: 2463: 2464: 2465: 2466: 2467: 2468: 2469: 2470: 2471: 2472: 2473: 2474: 2475: 2476: 2477: 2478: 2479: 2480: 2481: 2482: 2483: 2484: 2485: 2486: 2487: 2488: 2489: 2490: 2491: 2492: 2493: 2494: 2495: 2496: 2497: 2498: 2499: 2500: 2501: 2502: 2503: 2504: 2505: 2506: 2507: 2508: 2509: 2510: 2511: 2512: 2513: 2514: 2515: 2516: 2517: 2518: 2519: 2520: 2521: 2522: 2523: 2524: 2525: 2526: 2527: 2528: 2529: 2530: 2531: 2532: 2533: 2534: 2535: 2536: 2537: 2538: 2539: 2540: 2541: 2542: 2543: 2544: 2545: 2546: 2547: 2548: 2549: 2550: 2551: 2552: 2553: 2554: 2555: 2556: 2557: 2558: 2559: 2560: 2561: 2562: 2563: 2564: 2565: 2566: 2567: 2568: 2569: 2570: 2571: 2572: 2573: 2574: 2575: 2576: 2577: 2578: 2579: 2580: 2581: 2582: 2583: 2584: 2585: 2586: 2587: 2588: 2589: 2590: 2591: 2592: 2593: 2594: 2595: 2596: 2597: 2598: 2599: 2600: 2601: 2602: 2603: 2604: 2605: 2606: 2607: 2608: 2609: 2610: 2611: 2612: 2613: 2614: 2615: 2616: 2617: 2618: 2619: 2620: 2621: 2622: 2623: 2624: 2625: 2626: 2627: 2628: 2629: 2630: 2631: 2632: 2633: 2634: 2635: 2636: 2637: 2638: 2639: 2640: 2641: 2642: 2643: 2644: 2645: 2646: 2647: 2648: 2649: 2650: 2651: 2652: 2653: 2654: 2655: 2656: 2657: 2658: 2659: 2660: 2661: 2662: 2663: 2664: 2665: 2666: 2667: 2668: 2669: 2670: 2671: 2672: 2673: 2674: 2675: 2676: 2677: 2678: 2679: 2680: 2681: 2682: 2683: 2684: 2685: 2686: 2687: 2688: 2689: 2690: 2691: 2692: 2693: 2694: 2695: 2696: 2697: 2698: 2699: 2700: 2701: 2702: 2703: 2704: 2705: 2706: 2707: 2708: 2709: 2710: 2711: 2712: 2713: 2714: 2715: 2716: 2717: 2718: 2719: 2720: 2721: 2722: 2723: 2724: 2725: 2726: 2727: 2728: 2729: 2730: 2731: 2732: 2733: 2734: 2735: 2736: 2737: 2738: 2739: 2740: 2741: 2742: 2743: 2744: 2745: 2746: 2747: 2748: 2749: 2750: 2751: 2752: 2753: 2754: 2755: 2756: 2757: 2758: 2759: 2760: 2761: 2762: 2763: 2764: 2765: 2766: 2767: 2768: 2769: 2770: 2771: 2772: 2773: 2774: 2775: 2776: 2777: 2778: 2779: 2780: 2781: 2782: 2783: 2784: 2785: 2786: 2787: 2788: 2789: 2790: 2791: 2792: 2793: 2794: 2795: 2796: 2797: 2798: 2799: 2800: 2801: 2802: 2803: 2804: 2805: 2806: 2807: 2808: 2809: 2810: 2811: 2812: 2813: 2814: 2815: 2816: 2817: 2818: 2819: 2820: 2821: 2822: 2823: 2824: 2825: 2826: 2827: 2828: 2829: 2830: 2831: 2832: 2833: 2834: 2835: 2836: 2837: 2838: 2839: 2840: 2841: 2842: 2843: 2844: 2845: 2846: 2847: 2848: 2849: 2850: 2851: 2852: 2853: 2854: 2855: 2856: 2857: 2858: 2859: 2860: 2861: 2862: 2863: 2864: 2865: 2866: 2867: 2868: 2869: 2870: 2871: 2872: 2873: 2874: 2875: 2876: 2877: 2878: 2879: 2880: 2881: 2882: 2883: 2884: 2885: 2886: 2887: 2888: 2889: 2890: 2891: 2892: 2893: 2894: 2895: 2896: 2897: 2898: 2899: 2900: 2901: 2902: 2903: 2904: 2905: 2906: 2907: 2908: 2909: 2910: 2911: 2912: 2913: 2914: 2915: 2916: 2917: 2918: 2919: 2920: 2921: 2922: 2923: 2924: 2925: 2926: 2927: 2928: 2929: 2930: 2931: 2932: 2933: 2934: 2935: 2936: 2937: 2938: 2939: 2940: 2941: 2942: 2943: 2944: 2945: 2946: 2947: 2948: 2949: 2950: 2951: 2952: 2953: 2954: 2955: 2956: 2957: 2958: 2959: 2960: 2961: 2962: 2963: 2964: 2965: 2966: 2967: 2968: 2969: 2970: 2971: 2972: 2973: 2974: 2975: 2976: 2977: 2978: 2979: 2980: 2981: 2982: 2983: 2984: 2985: 2986: 2987: 2988: 2989: 2990: 2991: 2992: 2993: 2994: 2995: 2996: 2997: 2998: 2999: 3000: 3001: 3002: 3003: 3004: 3005: 3006: 3007: 3008: 3009: 3010: 3011: 3012: 3013: 3014: 3015: 3016: 3017: 3018: 3019: 3020: 3021: 3022: 3023: 3024: 3025: 3026: 3027: 3028: 3029: 3030: 3031: 3032: 3033: 3034: 3035: 3036: 3037: 3038: 3039: 3040: 3041: 3042: 3043: 3044: 3045: 3046: 3047: 3048: 3049: 3050: 3051: 3052: 3053: 3054: 3055: 3056: 3057: 3058: 3059: 3060: 3061: 3062: 3063: 3064: 3065: 3066: 3067: 3068: 3069: 3070: 3071: 3072: 3073: 3074: 3075: 3076: 3077: 3078: 3079: 3080: 3081: 3082: 3083: 3084: 3085: 3086: 3087: 3088: 3089: 3090: 3091: 3092: 3093: 3094: 3095: 3096: 3097: 3098: 3099: 3100: 3101: 3102: 3103: 3104: 3105: 3106: 3107: 3108: 3109: 3110: 3111: 3112: 3113: 3114: 3115: 3116: 3117: 3118: 3119: 3120: 3121: 3122: 3123: 3124: 3125: 3126: 3127: 3128: 3129: 3130: 3131: 3132: 3133: 3134: 3135: 3136: 3137: 3138: 3139: 3140: 3141: 3142: 3143: 3144: 3145: 3146: 3147: 3148: 3149: 3150: 3151: 3152: 3153: 3154: 3155: 3156: 3157: 3158: 3159: 3160: 3161: 3162: 3163: 3164: 3165: 3166: 3167: 3168: 3169: 3170: 3171: 3172: 3173: 3174: 3175: 3176: 3177: 3178: 3179: 3180: 3181: 3182: 3183: 3184: 3185: 3186: 3187: 3188: 3189: 3190: 3191: 3192: 3193: 3194: 3195: 3196: 3197: 3198: 3199: 3200: 3201: 3202: 3203: 3204: 3205: 3206: 3207: 3208: 3209: 3210: 3211: 3212: 3213: 3214: 3215: 3216: 3217: 3218: 3219: 3220: 3221: 3222: 3223: 3224: 3225: 3226: 3227: 3228: 3229: 3230: 3231: 3232: 3233: 3234: 3235: 3236: 3237: 3238: 3239: 3240: 3241: 3242: 3243: 3244: 3245: 3246: 3247: 3248: 3249: 3250: 3251: 3252: 3253: 3254: 3255: 3256: 3257: 3258: 3259: 3260: 3261: 3262: 3263: 3264: 3265: 3266: 3267: 3268: 3269: 3270: 3271: 3272: 3273: 3274: 3275: 3276: 3277: 3278: 3279: 3280: 3281: 3282: 3283: 3284: 3285: 3286: 3287: 3288: 3289: 3290: 3291: 3292: 3293: 3294: 3295: 3296: 3297: 3298: 3299: 3300: 3301: 3302: 3303: 3304: 3305: 3306: 3307: 3308: 3309: 3310: 3311: 3312: 3313: 3314: 3315: 3316: 3317: 3318: 3319: 3320: 3321: 3322: 3323: 3324: 3325: 3326: 3327: 3328: 3329: 3330: 3331: 3332: 3333: 3334: 3335: 3336: 3337: 3338: 3339: 3340: 3341: 3342: 3343: 3344: 3345: 3346: 3347: 3348: 3349: 3350: 3351: 3352: 3353: 3354: 3355: 3356: 3357: 3358: 3359: 3360: 3361: 3362: 3363: 3364: 3365: 3366: 3367: 3368: 3369: 3370: 3371: 3372: 3373: 3374: 3375: 3376: 3377: 3378: 3379: 3380: 3381: 3382: 3383: 3384: 3385: 3386: 3387: 3388: 3389: 3390: 3391: 3392: 3393: 3394: 3395: 3396: 3397: 3398: 3399: 3400: 3401: 3402: 3403: 3404: 3405: 3406: 3407: 3408: 3409: 3410: 3411: 3412: 3413: 3414: 3415: 3416: 3417: 3418: 3419: 3420: 3421: 3422: 3423: 3424: 3425: 3426: 3427: 3428: 3429: 3430: 3431: 3432: 3433: 3434: 3435: 3436: 3437: 3438: 3439: 3440: 3441: 3442: 3443: 3444: 3445: 3446: 3447: 3448: 3449: 3450: 3451: 3452: 3453: 3454: 3455: 3456: 3457: 3458: 3459: 3460: 3461: 3462: 3463: 3464: 3465: 3466: 3467: 3468: 3469: 3470: 3471: 3472: 3473: 3474: 3475: 3476: 3477: 3478: 3479: 3480: 3481: 3482: 3483: 3484: 3485: 3486: 3487: 3488: 3489: 3490: 3491: 3492: 3493: 3494: 3495: 3496: 3497: 3498: 3499: 3500: 3501: 3502: 3503: 3504: 3505: 3506: 3507: 3508: 3509: 3510: 3511: 3512: 3513: 3514: 3515: 3516: 3517: 3518: 3519: 3520: 3521: 3522: 3523: 3524: 3525: 3526: 3527: 3528: 3529: 3530: 3531: 3532: 3533: 3534: 3535: 3536: 3537: 3538: 3539: 3540: 3541: 3542: 3543: 3544: 3545: 3546: 3547: 3548: 3549: 3550: 3551: 3552: 3553: 3554: 3555: 3556: 3557: 3558: 3559: 3560: 3561: 3562: 3563: 3564: 3565: 3566: 3567: 3568: 3569: 3570: 3571: 3572: 3573: 3574: 3575: 3576: 3577: 3578: 3579: 3580: 3581: 3582: 3583: 3584: 3585: 3586: 3587: 3588: 3589: 3590: 3591: 3592: 3593: 3594: 3595: 3596: 3597: 3598: 3599: 3600: 3601: 3602: 3603: 3604: 3605: 3606: 3607: 3608: 3609: 3610: 3611: 3612: 3613: 3614: 3615: 3616: 3617: 3618: 3619: 3620: 3621: 3622: 3623: 3624: 3625: 3626: 3627: 3628: 3629: 3630: 3631: 3632: 3633: 3634: 3635: 3636: 3637: 3638: 3639: 3640: 3641: 3642: 3643: 3644: 3645: 3646: 3647: 3648: 3649: 3650: 3651: 3652: 3653: 3654: 3655: 3656: 3657: 3658: 3659: 3660: 3661: 3662: 3663: 3664: 3665: 3666: 3667: 3668: 3669: 3670: 3671: 3672: 3673: 3674: 3675: 3676: 3677: 3678: 3679: 3680: 3681: 3682: 3683: 3684: 3685: 3686: 3687: 3688: 3689: 3690: 3691: 3692: 3693: 3694: 3695: 3696: 3697: 3698: 3699: 3700: 3701: 3702: 3703: 3704: 3705: 3706: 3707: 3708: 3709: 3710: 3711: 3712: 3713: 3714: 3715: 3716: 3717: 3718: 3719: 3720: 3721: 3722: 3723: 3724: 3725: 3726: 3727: 3728: 3729: 3730: 3731: 3732: 3733: 3734: 3735: 3736: 3737: 3738: 3739: 3740: 3741: 3742: 3743: 3744: 3745: 3746: 3747: 3748: 3749: 3750: 3751: 3752: 3753: 3754: 3755: 3756: 3757: 3758: 3759: 3760: 3761: 3762: 3763: 3764: 3765: 3766: 3767: 3768: 3769: 3770: 3771: 3772: 3773: 3774: 3775: 3776: 3777: 3778: 3779: 3780: 3781: 3782: 3783: 3784: 3785: 3786: 3787: 3788: 3789: 3790: 3791: 3792: 3793: 3794: 3795: 3796: 3797: 3798: 3799: 3800: 3801: 3802: 3803: 3804: 3805: 3806: 3807: 3808: 3809: 3810: 3811: 3812: 3813: 3814: 3815: 3816: 3817: 3818: 3819: 3820: 3821: 3822: 3823: 3824: 3825: 3826: 3827: 3828: 3829: 3830: 3831: 3832: 3833: 3834: 3835: 3836: 3837: 3838: 3839: 3840: 3841: 3842: 3843: 3844: 3845: 3846: 3847: 3848: 3849: 3850: 3851: 3852: 3853: 3854: 3855: 3856: 3857: 3858: 3859: 3860: 3861: 3862: 3863: 3864: 3865: 3866: 3867: 3868: 3869: 3870: 3871: 3872: 3873: 3874: 3875: 3876: 3877: 3878: 3879: 3880: 3881: 3882: 3883: 3884: 3885: 3886: 3887: 3888: 3889: 3890: 3891: 3892: 3893: 3894: 3895: 3896: 3897: 3898: 3899: 3900: 3901: 3902: 3903: 3904: 3905: 3906: 3907: 3908: 3909: 3910: 3911: 3912: 3913: 3914: 3915: 3916: 3917: 3918: 3919: 3920: 3921: 3922: 3923: 3924: 3925: 3926: 3927: 3928: 3929: 3930: 3931: 3932: 3933: 3934: 3935: 3936: 3937: 3938: 3939: 3940: 3941: 3942: 3943: 3944: 3945: 3946: 3947: 3948: 3949: 3950: 3951: 3952: 3953: 3954: 3955: 3956: 3957: 3958: 3959: 3960: 3961: 3962: 3963: 3964: 3965: 3966: 3967: 3968: 3969: 3970: 3971: 3972: 3973: 3974: 3975: 3976: 3977: 3978: 3979: 3980: 3981: 3982: 3983: 3984: 3985: 3986: 3987: 3988: 3989: 3990: 3991: 3992: 3993: 3994: 3995: 3996: 3997: 3998: 3999: 4000: 4001: 4002: 4003: 4004: 4005: 4006: 4007: 4008: 4009: 4010: 4011: 4012: 4013: 4014: 4015: 4016: 4017: 4018: 4019: 4020: 4021: 4022: 4023: 4024: 4025: 4026: 4027: 4028: 4029: 4030: 4031: 4032: 4033: 4034: 4035: 4036: 4037: 4038: 4039: 4040: 4041: 4042: 4043: 4044: 4045: 4046: 4047: 4048: 4049: 4050: 4051: 4052: 4053: 4054: 4055: 4056: 4057: 4058: 4059: 4060: 4061: 4062: 4063: 4064: 4065: 4066: 4067: 4068: 4069: 4070: 4071: 4072: 4073: 4074: 4075: 4076: 4077: 4078: 4079: 4080: 4081: 4082: 4083: 4084: 4085: 4086: 4087: 4088: 4089: 4090: 4091: 4092: 4093: 4094: 4095: 4096: 4097: 4098: 4099: 4100: 4101: 4102: 4103: 4104: 4105: 4106: 4107: 4108: 4109: 4110: 4111: 4112: 4113: 4114: 4115: 4116: 4117: 4118: 4119: 4120: 4121: 4122: 4123: 4124: 4125: 4126: 4127: 4128: 4129: 4130: 4131: 4132: 4133: 4134: 4135: 4136: 4137: 4138: 4139: 4140: 4141: 4142: 4143: 4144: 4145: 4146: 4147: 4148: 4149: 4150: 4151: 4152: 4153: 4154: 4155: 4156: 4157: 4158: 4159: 4160: 4161: 4162: 4163: 4164: 4165: 4166: 4167: 4168: 4169: 4170: 4171: 4172: 4173: 4174: 4175: 4176: 4177: 4178: 4179: 4180: 4181: 4182: 4183: 4184: 4185: 4186: 4187: 4188: 4189: 4190: 4191: 4192: 4193: 4194: 4195: 4196: 4197: 4198: 4199: 4200: 4201: 4202: 4203: 4204: 4205: 4206: 4207: 4208: 4209: 4210: 4211: 4212: 4213: 4214: 4215: 4216: 4217: 4218: 4219: 4220: 4221: 4222: 4223: 4224: 4225: 4226: 4227: 4228: 4229: 4230: 4231: 4232: 4233: 4234: 4235: 4236: 4237: 4238: 4239: 4240: 4241: 4242: 4243: 4244: 4245: 4246: 4247: 4248: 4249: 4250: 4251: 4252: 4253: 4254: 4255: 4256: 4257: 4258: 4259: 4260: 4261: 4262: 4263: 4264: 4265: 4266: 4267: 4268: 4269: 4270: 4271: 4272: 4273: 4274: 4275: 4276: 4277: 4278: 4279: 4280: 4281: 4282: 4283: 4284: 4285: 4286: 4287: 4288: 4289: 4290: 4291: 4292: 4293: 4294: 4295: 4296: 4297: 4298: 4299: 4300: 4301: 4302: 4303: 4304: 4305: 4306: 4307: 4308: 4309: 4310: 4311: 4312: 4313: 4314: 4315: 4316: 4317: 4318: 4319: 4320: 4321: 4322: 4323: 4324: 4325: 4326: 4327: 4328: 4329: 4330: 4331: 4332: 4333: 4334: 4335: 4336: 4337: 4338: 4339: 4340: 4341: 4342: 4343: 4344: 4345: 4346: 4347: 4348: 4349: 4350: 4351: 4352: 4353: 4354: 4355: 4356: 4357: 4358: 4359: 4360: 4361: 4362: 4363: 4364: 4365: 4366: 4367: 4368: 4369: 4370: 4371: 4372: 4373: 4374: 4375: 4376: 4377: 4378: 4379: 4380: 4381: 4382: 4383: 4384: 4385: 4386: 4387: 4388: 4389: 4390: 4391: 4392: 4393: 4394: 4395: 4396: 4397: 4398: 4399: 4400: 4401: 4402: 4403: 4404: 4405: 4406: 4407: 4408: 4409: 4410: 4411: 4412: 4413: 4414: 4415: 4416: 4417: 4418: 4419: 4420: 4421: 4422: 4423: 4424: 4425: 4426: 4427: 4428: 4429: 4430: 4431: 4432: 4433: 4434: 4435: 4436: 4437: 4438: 4439: 4440: 4441: 4442: 4443: 4444: 4445: 4446: 4447: 4448: 4449: 4450: 4451: 4452: 4453: 4454: 4455: 4456: 4457: 4458: 4459: 4460: 4461: 4462: 4463: 4464: 4465: 4466: 4467: 4468: 4469: 4470: 4471: 4472: 4473: 4474: 4475: 4476: 4477: 4478: 4479: 4480: 4481: 4482: 4483: 4484: 4485: 4486: 4487: 4488: 4489: 4490: 4491: 4492: 4493: 4494: 4495: 4496: 4497: 4498: 4499: 4500: 4501: 4502: 4503: 4504: 4505: 4506: 4507: 4508: 4509: 4510: 4511: 4512: 4513: 4514: 4515: 4516: 4517: 4518: 4519: 4520: 4521: 4522: 4523: 4524: 4525: 4526: 4527: 4528: 4529: 4530: 4531: 4532: 4533: 4534: 4535: 4536: 4537: 4538: 4539: 4540: 4541: 4542: 4543: 4544: 4545: 4546: 4547: 4548: 4549: 4550: 4551: 4552: 4553: 4554: 4555: 4556: 4557: 4558: 4559: 4560: 4561: 4562: 4563: 4564: 4565: 4566: 4567: 4568: 4569: 4570: 4571: 4572: 4573: 4574: 4575: 4576: 4577: 4578: 4579: 4580: 4581: 4582: 4583: 4584: 4585: 4586: 4587: 4588: 4589: 4590: 4591: 4592: 4593: 4594: 4595: 4596: 4597: 4598: 4599: 4600: 4601: 4602: 4603: 4604: 4605: 4606: 4607: 4608: 4609: 4610: 4611: 4612: 4613: 4614: 4615: 4616: 4617: 4618: 4619: 4620: 4621: 4622: 4623: 4624: 4625: 4626: 4627: 4628: 4629: 4630: 4631: 
<?php
/**
 * Functions used to display content in themes.
 * @package functions
 * @subpackage template-functions
 */
// force UTF-8 Ø

require_once(dirname(__FILE__) . '/functions.php');
if (!defined('SEO_FULLWEBPATH')) {
    define('SEO_FULLWEBPATH', FULLWEBPATH);
    define('SEO_WEBPATH', WEBPATH);
}

//******************************************************************************
//*** Template Functions *******************************************************
//******************************************************************************

/* * * Generic Helper Functions ************ */
/* * *************************************** */

/**
 * Returns the zenphoto version string
 */
function getVersion() {
    return ZENPHOTO_VERSION;
}

/**
 * Prints the zenphoto version string
 */
function printVersion() {
    echo getVersion();
}

/**
 * Print any Javascript required by zenphoto.
 */
function printZenJavascripts() {
    global $_zp_current_album;
    ?>
    <script type="text/javascript" src="<?php echo WEBPATH . "/" . ZENFOLDER; ?>/js/jquery.js"></script>
    <?php
    if(zp_loggedin()) {
        ?>
        <script type="text/javascript" src="<?php echo WEBPATH . "/" . ZENFOLDER; ?>/js/zenphoto.js"></script>
        <link rel="stylesheet" href="<?php echo WEBPATH . '/' . ZENFOLDER; ?>/admintoolbox.css" type="text/css" />
        <?php
    }
}

/**
 * Prints the clickable drop down toolbox on any theme page with generic admin helpers
 *
 */
function adminToolbox() {
    global $_zp_current_album, $_zp_current_image, $_zp_current_search, $_zp_gallery_page, $_zp_gallery, $_zp_current_admin_obj, $_zp_loggedin, $_zp_conf_vars;
    if (zp_loggedin()) {
        $zf = FULLWEBPATH . "/" . ZENFOLDER;
        $page = getCurrentPage();
        ob_start();
        ?>
        <script type="text/javascript">
            // <!-- <![CDATA[
            var deleteAlbum1 = "<?php echo gettext("Are you sure you want to delete this item?"); ?>";
            var deleteAlbum2 = "<?php echo gettext("Are you Absolutely positively sure you want to delete this item? THIS CANNOT BE UNDONE!"); ?>";
            function newAlbum(folder, albumtab) {
                var album = prompt('<?php echo gettext('New album name?'); ?>', '<?php echo gettext('new album'); ?>');
                if (album) {
                    launchScript('<?php echo $zf; ?>/admin-edit.php', ['action=newalbum', 'folder=' + encodeURIComponent(folder), 'name=' + encodeURIComponent(album), 'albumtab=' + albumtab, 'XSRFToken=<?php echo getXSRFToken('newalbum'); ?>']);
                }
            }
            // ]]> -->
        </script>
        <div id="zp__admin_module">
            <div id="zp__admin_info">
                <span class="zp_logo">ZP</span>
                <span class="zp_user"> <?php echo $_zp_current_admin_obj->getUser(); ?>
                    <?php
                    if(array_key_exists('site_upgrade_state', $_zp_conf_vars)) {
                        if($_zp_conf_vars['site_upgrade_state'] == 'closed_for_test') {
                            echo ' | <span class="zp_sitestatus">' . gettext('Test mode') . '</span>';
                        }
                    }
                    ?>
                </span>
            </div>
            <button type="button" id="zp__admin_link" onclick="javascript:toggle('zp__admin_data');">
                <?php echo gettext('Admin'); ?>
            </button>
            <div id="zp__admin_data" style="display: none;">
                <ul>
                <?php
                $outputA = ob_get_contents();
                ob_end_clean();
                ob_start();

                if (zp_loggedin(OVERVIEW_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin.php', gettext("Overview"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                if (zp_loggedin(UPLOAD_RIGHTS | FILES_RIGHTS | THEMES_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-upload.php', gettext("Upload"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                if (zp_loggedin(ALBUM_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-edit.php', gettext("Albums"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                zp_apply_filter('admin_toolbox_global', $zf);

                if (zp_loggedin(TAGS_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-tags.php', gettext("Tags"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                if (zp_loggedin(USER_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-users.php', gettext("Users"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                if (zp_loggedin(OPTIONS_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-options.php?tab=general', gettext("Options"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                if (zp_loggedin(THEMES_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-themes.php', gettext("Themes"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }
                if (zp_loggedin(ADMIN_RIGHTS)) {
                    ?>
                    <li>
                        <?php printLinkHTML($zf . '/admin-plugins.php', gettext("Plugins"), NULL, NULL, NULL); ?>
                    </li>
                    <li>
                        <?php printLinkHTML($zf . '/admin-logs.php', gettext("Logs"), NULL, NULL, NULL); ?>
                    </li>
                    <?php
                }

                $gal = getOption('custom_index_page');
                if (empty($gal) || !file_exists(SERVERPATH . '/' . THEMEFOLDER . '/' . $_zp_gallery->getCurrentTheme() . '/' . internalToFilesystem($gal) . '.php')) {
                    $gal = 'index.php';
                } else {
                    $gal .= '.php';
                }
                $inImage = false;
                switch ($_zp_gallery_page) {
                    case 'index.php':
                    case $gal:
                        // script is either index.php or the gallery index page
                        if (zp_loggedin(ADMIN_RIGHTS)) {
                            ?>
                            <li>
                                <?php printLinkHTML($zf . '/admin-edit.php?page=edit', gettext("Sort Gallery"), NULL, NULL, NULL); ?>
                            </li>
                            <?php
                        }
                        if (zp_loggedin(MANAGE_ALL_ALBUM_RIGHTS)) {
                            // admin has upload rights, provide an upload link for a new album
                            ?>
                            <li>
                                <a href="javascript:newAlbum('',true);"><?php echo gettext("New Album"); ?></a>
                            </li>
                            <?php
                        }
                        if ($_zp_gallery_page == 'index.php') {
                            $redirect = '';
                        } else {
                            $redirect = "&amp;p=" . urlencode(stripSuffix($_zp_gallery_page));
                        }
                        if ($page > 1) {
                            $redirect .= "&amp;page=$page";
                        }
                        zp_apply_filter('admin_toolbox_gallery', $zf);
                        break;
                    case 'image.php':
                        $inImage = true; // images are also in albums[sic]
                    case 'album.php':
                        // script is album.php
                        $albumname = $_zp_current_album->name;
                        if ($_zp_current_album->isMyItem(ALBUM_RIGHTS)) {
                            // admin is empowered to edit this album--show an edit link
                            ?>
                            <li>
                                <?php printLinkHTML($zf . '/admin-edit.php?page=edit&album=' . pathurlencode($_zp_current_album->name), gettext('Edit album'), NULL, NULL, NULL); ?>
                            </li>
                            <?php
                            if (!$_zp_current_album->isDynamic()) {
                                if ($_zp_current_album->getNumAlbums()) {
                                    ?>
                                    <li>
                                        <?php printLinkHTML($zf . '/admin-edit.php?page=edit&album=' . pathurlencode($albumname) . '&tab=subalbuminfo', gettext("Sort subalbums"), NULL, NULL, NULL); ?>
                                    </li>
                                    <?php
                                }
                                if ($_zp_current_album->getNumImages() > 0) {
                                    ?>
                                    <li>
                                        <?php printLinkHTML($zf . '/admin-albumsort.php?page=edit&album=' . pathurlencode($albumname) . '&tab=sort', gettext("Sort images"), NULL, NULL, NULL); ?>
                                    </li>
                                    <?php
                                }
                            }
                            // and a delete link
                            ?>
                            <li>
                                <a href="javascript:confirmDeleteAlbum('<?php echo $zf; ?>/admin-edit.php?page=edit&amp;action=deletealbum&amp;album=<?php echo urlencode(pathurlencode($albumname)) ?>&amp;XSRFToken=<?php echo getXSRFToken('delete'); ?>');"
                                     title="<?php echo gettext('Delete the album'); ?>"><?php echo gettext('Delete album'); ?></a>
                            </li>
                            <?php
                        }
                        if ($_zp_current_album->isMyItem(UPLOAD_RIGHTS) && !$_zp_current_album->isDynamic()) {
                            // provide an album upload link if the admin has upload rights for this album and it is not a dynamic album
                            ?>
                            <li>
                                <?php printLinkHTML($zf . '/admin-upload.php?album=' . pathurlencode($albumname), gettext("Upload Here"), NULL, NULL, NULL); ?>
                            </li>
                            <li>
                                <a href="javascript:newAlbum('<?php echo pathurlencode($albumname); ?>',true);"><?php echo gettext("New Album Here"); ?></a>
                            </li>
                            <?php
                        }
                        zp_apply_filter('admin_toolbox_album', $albumname, $zf);
                        if ($inImage) {
                            // script is image.php
                            $imagename = $_zp_current_image->filename;
                                if ($_zp_current_album->isMyItem(ALBUM_RIGHTS)) {
                                    if ($_zp_current_album->isDynamic()) { // get folder of the corresponding static album
                                        $albumobj = $_zp_current_image->getAlbum();
                                        $albumname = $albumobj->name;
                                    } else {
                                        $delete_image = gettext("Are you sure you want to delete this image? THIS CANNOT BE UNDONE!");
                                        // if admin has edit rights on this album, provide a delete link for the image.
                                        ?>
                                        <li>
                                            <a href="javascript:confirmDelete('<?php echo $zf; ?>/admin-edit.php?page=edit&amp;action=deleteimage&amp;album=<?php echo urlencode(pathurlencode($albumname)); ?>&amp;image=<?php echo urlencode($imagename); ?>&amp;XSRFToken=<?php echo getXSRFToken('delete'); ?>','<?php echo $delete_image; ?>');"
                                                 title="<?php echo gettext("Delete the image"); ?>"><?php echo gettext("Delete image"); ?></a>
                                        </li>
                                        <?php
                                    }
                                    ?>
                                    <li>
                                        <a href="<?php echo $zf; ?>/admin-edit.php?page=edit&amp;album=<?php echo pathurlencode($albumname); ?>&amp;singleimage=<?php echo urlencode($imagename); ?>&amp;tab=imageinfo&amp;nopagination"
                                             title="<?php echo gettext('Edit image'); ?>"><?php echo gettext('Edit image'); ?></a>
                                    </li>
                                    <?php
                                // set return to this image page
                                zp_apply_filter('admin_toolbox_image', $albumname, $imagename, $zf);
                            }
                            $redirect = "&amp;album=" . html_encode(pathurlencode($albumname)) . "&amp;image=" . urlencode($imagename);
                        } else {
                            // set the return to this album/page
                            $redirect = "&amp;album=" . html_encode(pathurlencode($albumname));
                            if ($page > 1) {
                                $redirect .= "&amp;page=$page";
                            }
                        }
                        break;
                    case 'search.php':
                        $words = $_zp_current_search->getSearchWords();
                        if (!empty($words)) {
                            // script is search.php with a search string
                            if (zp_loggedin(UPLOAD_RIGHTS)) {
                                $link = $zf . '/admin-dynamic-album.php?' . substr($_zp_current_search->getSearchParams(), 1);
                                // if admin has edit rights allow him to create a dynamic album from the search
                                ?>
                                <li>
                                    <a href="<?php echo $link; ?>" title="<?php echo gettext('Create an album from the search'); ?>" ><?php echo gettext('Create Album'); ?></a>
                                </li>
                                <?php
                            }
                            zp_apply_filter('admin_toolbox_search', $zf);
                        }
                        $redirect = "&amp;p=search" . $_zp_current_search->getSearchParams() . "&amp;page=$page";
                        break;
                    default:
                        // arbitrary custom page
                        $gal = stripSuffix($_zp_gallery_page);
                        $redirect = "&amp;p=" . urlencode($gal);
                        if ($page > 1) {
                            $redirect .= "&amp;page=$page";
                        }
                        $redirect = zp_apply_filter('admin_toolbox_' . $gal, $redirect, $zf);
                        break;
                }
                $redirect = zp_apply_filter('admin_toolbox_close', $redirect, $zf);
                if ($_zp_current_admin_obj->logout_link) {
                    // logout link
                    $sec = (int) ((SERVER_PROTOCOL == 'https') & true);
                    $link = SEO_FULLWEBPATH . '/index.php?logout=' . $sec . $redirect;
                    ?>
                    <li>
                        <?php printLinkHTML($link, gettext("Logout"), gettext("Logout"), null, null); ?>
                    </li>
                    <?php
                }
                $outputB = ob_get_contents();
                ob_end_clean();
                if ($outputB) {
                    echo $outputA . $outputB;
                    ?>
                </ul>
            </div>
        </div>
        <?php
        }
    }
}

//*** Gallery Index (album list) Context ***
//******************************************

/**
 * Returns the raw title of the gallery.
 *
 * @return string
 */
function getGalleryTitle() {
    global $_zp_gallery;
    return $_zp_gallery->getTitle();
}

/**
 * Returns a text-only title of the gallery.
 *
 * @return string
 */
function getBareGalleryTitle() {
    return getBare(getGalleryTitle());
}

/**
 * Prints the title of the gallery.
 */
function printGalleryTitle() {
    echo html_encodeTagged(getGalleryTitle());
}

function printBareGalleryTitle() {
    echo html_encode(getBareGalleryTitle());
}

/**
 * Function to create the page title to be used within the html <head> <title></title> element.
 * Usefull if you use one header.php for the header of all theme pages instead of individual ones on the theme pages
 * It returns the title and site name in reversed breadcrumb order:
 * <title of current page> | <parent item if present> | <gallery title>
 * It supports standard gallery pages as well a custom and Zenpage news articles, categories and pages.
 *
 * @param string $separator How you wish the parts to be separated
 * @param bool $listparentalbums If the parent albums should be printed in reversed order before the current
 * @param bool $listparentpage If the parent Zenpage pages should be printed in reversed order before the current page
 */
function getHeadTitle($separator = ' | ', $listparentalbums = true, $listparentpages = true) {
    global $_zp_gallery, $_zp_current_album, $_zp_current_image, $_zp_current_zenpage_news, $_zp_current_zenpage_page, $_zp_gallery_page, $_zp_current_category, $_zp_page, $_myFavorites;
    $mainsitetitle = html_encode(getBare(getMainSiteName()));
    $separator = html_encode($separator);
    if ($mainsitetitle) {
        $mainsitetitle = $separator . $mainsitetitle;
    }
    $gallerytitle = html_encode(getBareGalleryTitle());
    if ($_zp_page > 1) {
        $pagenumber = ' (' . $_zp_page . ')';
    } else {
        $pagenumber = '';
    }
    switch ($_zp_gallery_page) {
        case 'index.php':
            return $gallerytitle . $mainsitetitle . $pagenumber;
            break;
        case 'album.php':
        case 'image.php':
            if ($listparentalbums) {
                $parents = getParentAlbums();
                $parentalbums = '';
                if (count($parents) != 0) {
                    $parents = array_reverse($parents);
                    foreach ($parents as $parent) {
                        $parentalbums .= html_encode(getBare($parent->getTitle())) . $separator;
                    }
                }
            } else {
                $parentalbums = '';
            }
            $albumtitle = html_encode(getBareAlbumTitle()) . $pagenumber . $separator . $parentalbums . $gallerytitle . $mainsitetitle;
            switch ($_zp_gallery_page) {
                case 'album.php':
                    return $albumtitle;
                    break;
                case 'image.php':
                    return html_encode(getBareImageTitle()) . $separator . $albumtitle;
                    break;
            }
            break;
        case 'news.php':
            if (function_exists("is_NewsArticle")) {
                if (is_NewsArticle()) {
                    return html_encode(getBareNewsTitle()) . $pagenumber . $separator . gettext('News') . $separator . $gallerytitle . $mainsitetitle;
                } else if (is_NewsCategory()) {
                    return html_encode(getBare($_zp_current_category->getTitle())) . $pagenumber . $separator . gettext('News') . $separator . $gallerytitle . $mainsitetitle;
                } else {
                    return gettext('News') . $pagenumber . $separator . $gallerytitle . $mainsitetitle;
                }
            }
            break;
        case 'pages.php':
            if ($listparentpages) {
                $parents = $_zp_current_zenpage_page->getParents();
                $parentpages = '';
                if (count($parents) != 0) {
                    $parents = array_reverse($parents);
                    foreach ($parents as $parent) {
                        $obj = new ZenpagePage($parent);
                        $parentpages .= html_encode(getBare($obj->getTitle())) . $separator;
                    }
                }
            } else {
                $parentpages = '';
            }
            return html_encode(getBarePageTitle()) . $pagenumber . $separator . $parentpages . $gallerytitle . $mainsitetitle;
            break;
        case '404.php':
            return gettext('Object not found') . $separator . $gallerytitle . $mainsitetitle;
            break;
        default: // for all other possible static custom pages
            $custompage = stripSuffix($_zp_gallery_page);
            $standard = array(
                    'gallery' => gettext('Gallery'), 
                    'contact' => gettext('Contact'), 
                    'register' => gettext('Register'), 
                    'search' => gettext('Search'), 
                    'archive' => gettext('Archive view'), 
                    'password' => gettext('Password required'));
            if (is_object($_myFavorites)) {
                $standard['favorites'] = gettext('My favorites');
            }
            if (array_key_exists($custompage, $standard)) {
                return $standard[$custompage] . $pagenumber . $separator . $gallerytitle . $mainsitetitle;
            } else {
                return $custompage . $pagenumber . $separator . $gallerytitle . $mainsitetitle;
            }
            break;
    }
}

/**
 * Function to print the html <title>title</title> within the <head> of a html page based on the current theme page
 * Usefull if you use one header.php for the header of all theme pages instead of individual ones on the theme pages
 * It prints the title and site name including the <title> tag in reversed breadcrumb order:
 * <title><title of current page> | <parent item if present> | <gallery title></title>
 * It supports standard gallery pages as well a custom and Zenpage news articles, categories and pages.
 *
 * @param string $separator How you wish the parts to be separated
 * @param bool $listparentalbums If the parent albums should be printed in reversed order before the current
 * @param bool $listparentpage If the parent Zenpage pages should be printed in reversed order before the current page
 */
function printHeadTitle($separator = ' | ', $listparentalbums = true, $listparentpages = true) {
    echo '<title>' . getHeadTitle($separator, $listparentalbums, $listparentpages) . '</title>';
}

/**
 * Returns the raw description of the gallery.
 *
 * @return string
 */
function getGalleryDesc() {
    global $_zp_gallery;
    return $_zp_gallery->getDesc();
}

/**
 * Returns a text-only description of the gallery.
 *
 * @return string
 */
function getBareGalleryDesc() {
    return getBare(getGalleryDesc());
}

/**
 * Prints the description of the gallery.
 */
function printGalleryDesc() {
    echo html_encodeTagged(getGalleryDesc());
}

function printBareGalleryDesc() {
    echo html_encode(getBareGalleryDesc());
}

/**
 * Returns the name of the main website as set by the "Website Title" option
 * on the gallery options tab. Use this if Zenphoto is only a part of your website.
 *
 * @return string
 */
function getMainSiteName() {
    global $_zp_gallery;
    return $_zp_gallery->getWebsiteTitle();
}

/**
 * Returns the URL of the main website as set by the "Website URL" option
 * on the gallery options tab. Use this if Zenphoto is only a part of your website.
 *
 * @return string
 */
function getMainSiteURL() {
    global $_zp_gallery;
    return $_zp_gallery->getWebsiteURL();
}

/**
 * Returns the URL of the main gallery index page. If a custom index page is set this returns that page.
 * So this is not necessarily the home page of the site!
 * @return string
 */
function getGalleryIndexURL() {
    global $_zp_current_album, $_zp_gallery_page;
    $page = 1;
    if (in_context(ZP_ALBUM) && $_zp_gallery_page != 'index.php') {
        $album = getUrAlbum($_zp_current_album);
        $page = $album->getGalleryPage();
    }
    if (!$link = getCustomGalleryIndexURL($page)) {
        $link = getStandardGalleryIndexURL($page);
    }
    return zp_apply_filter('getLink', $link, 'index.php', NULL);
}

/**
 * Returns the url to the standard gallery index.php page
 *
 * @see getGalleryIndexURL()
 *
 * @param int $page Pagenumber to append
 * @param bool $webpath host path to be prefixed. If "false" is passed you will get a localized "WEBPATH"
 * @return string
 */
function getStandardGalleryIndexURL($page = 1, $webpath = null) {
    if ($page > 1) {
        return rewrite_path('/' . _PAGE_ . '/' . $page . '/', "/index.php?" . "page=" . $page, $webpath);
    } else {
        if (is_null($webpath)) {
            if (class_exists('seo_locale')) {
                $webpath = seo_locale::localePath();
            } else {
                $webpath = WEBPATH;
            }
        }
        return $webpath . "/";
    }
}

/**
 * Gets the custom gallery index url if one is set, otherwise false
 *
 * @see getGalleryIndexURL()
 *
 * @global array $_zp_conf_vars
 * @param int $page Pagenumber for pagination
 * @param bool $webpath host path to be prefixed. If "false" is passed you will get a localized "WEBPATH"
 * @return string
 */
function getCustomGalleryIndexURL($page = 1, $webpath = null) {
    $custom_index = getOption('custom_index_page');
    if ($custom_index) {
        $link = getCustomPageURL($custom_index, '', $webpath);
        if ($page > 1) {
            if (MOD_REWRITE) {
                $link .= $page . '/';
            } else {
                $link .= "&amp;page=" . $page;
            }
        }
        return $link;
    }
    return false;
}

/**
 * Returns the name to the individual custom gallery index page name if set,
 * otherwise returns generic custom gallery page "gallery.php" that is widely supported by themes
 * If you need to check if there is an indovidual custom_index_page set use
 * `getOption('custom_index_page')` or `getCustomGalleryIndexURL()`
 *
 * @return string
 */
function getCustomGalleryIndexPage() {
    $custom_index = getOption('custom_index_page');
    if ($custom_index) {
        return $custom_index . '.php';
    }
    return 'gallery.php';
}

/**
 * If a custom gallery index page is set this first prints a link to the actual site index (home page = index.php)
 * followed by the gallery index page link. Otherwise just the gallery index link
 *
 * @since 1.4.9
 * @param string $after Text to append after and outside the link for breadcrumbs
 * @param string $text Name of the link, if NULL "Gallery" is used
 * @param bool $printHomeURL In case of a custom gallery index, display breadcrumb with home link (default is true)
 */
function printGalleryIndexURL($after = NULL, $text = NULL, $printHomeURL = true) {
    global $_zp_gallery_page;
    if (is_null($text)) {
        $text = gettext('Gallery');
    }
    $customgalleryindex = getOption('custom_index_page');
    if ($customgalleryindex && $printHomeURL) {
        printSiteHomeURL($after);
    }
    if ($_zp_gallery_page == getCustomGalleryIndexPage()) {
        $after = NULL;
    }
    if (!$customgalleryindex || ($customgalleryindex && in_array($_zp_gallery_page, array('image.php', 'album.php', getCustomGalleryIndexPage())))) {
        printLinkHTML(getGalleryIndexURL(), $text, $text, 'galleryindexurl');
        echo $after;
    }
}


/**
 * Returns the home page link (WEBPATH) to the Zenphoto theme index.php page
 * Use in breadcrumbs if the theme uses a custom gallery index page so the gallery is not the site's home page
 *
 * @since 1.4.9
 * @global string $_zp_gallery_page
 * @return string
 */
function getSiteHomeURL() {
    return WEBPATH . '/';
}

/**
 * Prints the home page link (WEBPATH with trailing slash) to a Zenphoto theme index.php page
 * Use in breadcrumbs if the theme uses a custom gallery index page so the gallery is not the site's home page
 *
 * @param string $after Text after and outside the link for breadcrumbs
 * @param string $text Text of the link, if NULL "Home"
 */
function printSiteHomeURL($after = NULL, $text = NULL) {
    global $_zp_gallery_page;
    if ($_zp_gallery_page == 'index.php') {
        $after = '';
    }
    if (is_null($text)) {
        $text = gettext('Home');
    }
    printLinkHTML(getSiteHomeURL(), $text, $text, 'homeurl');
    echo $after;
}

/**
 * If the privacy page url option is set this prints a link to it
 * @param string $before To print before the link
 * @param string $after To print after the link
 */
function printPrivacyPageLink($before = null, $after = null) {
    $data = getDataUsageNotice();
    if (!empty($data['url'])) {
        echo $before;
        printLinkHTML($data['url'], $data['linktext'], $data['linktext'], null, null);
        echo $after;
    }
}

/**
 * Returns the number of albums.
 *
 * @return int
 */
function getNumAlbums() {
    global $_zp_gallery, $_zp_current_album, $_zp_current_search;
    if (in_context(ZP_SEARCH) && is_null($_zp_current_album)) {
        return $_zp_current_search->getNumAlbums();
    } else if (in_context(ZP_ALBUM)) {
        return $_zp_current_album->getNumAlbums();
    } else {
        return $_zp_gallery->getNumAlbums();
    }
}

/**
 * Returns the name of the currently active theme
 *
 * @return string
 */
function getCurrentTheme() {
    global $_zp_gallery;
    return $_zp_gallery->getCurrentTheme();
}

/* * * Album AND Gallery Context *********** */
/* * *************************************** */

/**
 * WHILE next_album(): context switches to Album.
 * If we're already in the album context, this is a sub-albums loop, which,
 * quite simply, changes the source of the album list.
 * Switch back to the previous context when there are no more albums.

 * Returns true if there are albums, false if none
 *
 * @param bool $all true to go through all the albums
 * @param bool $mine override the password checks
 * @return bool
 * @since 0.6
 */
function next_album($all = false, $mine = NULL) {
    global $_zp_albums, $_zp_gallery, $_zp_current_album, $_zp_page, $_zp_current_album_restore, $_zp_current_search;
    if (is_null($_zp_albums)) {
        if (in_context(ZP_SEARCH)) {
            $_zp_albums = $_zp_current_search->getAlbums($all ? 0 : $_zp_page, NULL, NULL, true, $mine);
        } else if (in_context(ZP_ALBUM)) {
            $_zp_albums = $_zp_current_album->getAlbums($all ? 0 : $_zp_page, NULL, NULL, true, $mine);
        } else {
            $_zp_albums = $_zp_gallery->getAlbums($all ? 0 : $_zp_page, NULL, NULL, true, $mine);
        }
        if (empty($_zp_albums)) {
            return NULL;
        }
        $_zp_current_album_restore = $_zp_current_album;
        $_zp_current_album = newAlbum(array_shift($_zp_albums), true, true);
        save_context();
        add_context(ZP_ALBUM);
        return true;
    } else if (empty($_zp_albums)) {
        $_zp_albums = NULL;
        $_zp_current_album = $_zp_current_album_restore;
        restore_context();
        return false;
    } else {
        $_zp_current_album = newAlbum(array_shift($_zp_albums), true, true);
        return true;
    }
}

/**
 * Returns the number of the current page without printing it.
 *
 * @return int
 */
function getCurrentPage() {
    global $_zp_page;
    return $_zp_page;
}

/**
 * Gets an array of the album ids of all accessible albums (publich or user dependend)
 *
 * @param object $obj from whence to get the albums
 * @param array $albumlist collects the list
 * @param bool $scan force scan for new images in the album folder
 */
function getAllAccessibleAlbums($obj, &$albumlist, $scan) {
    global $_zp_gallery;
    $locallist = $obj->getAlbums();
 foreach ($locallist as $folder) {
        $album = newAlbum($folder);
        If (!$album->isDynamic() && $album->checkAccess()) {
            if ($scan)
                $album->getImages();
            $albumlist[] = $album->getID();
            getAllAccessibleAlbums($album, $albumlist, $scan);
        }
    }
}

/**
 * Returns the number of pages for the current object
 *
 * @param bool $_oneImagePage set to true if your theme collapses all image thumbs
 * or their equivalent to one page. This is typical with flash viewer themes
 *
 * @return int
 */
function getTotalPages($_oneImagePage = false) {
    global $_zp_gallery, $_zp_current_album, $_firstPageImages, $_zp_zenpage, $_zp_current_category;
    if (in_context(ZP_ALBUM | ZP_SEARCH)) {
        $albums_per_page = max(1, getOption('albums_per_page'));
        $pageCount = (int) ceil(getNumAlbums() / $albums_per_page);
        $imageCount = getNumImages();
        if ($_oneImagePage) {
            if ($_oneImagePage === true) {
                $imageCount = min(1, $imageCount);
            } else {
                $imageCount = 0;
            }
        }
        $images_per_page = max(1, getOption('images_per_page'));
        $pageCount = ($pageCount + ceil(($imageCount - $_firstPageImages) / $images_per_page));
        return $pageCount;
    } else if (get_context() == ZP_INDEX) {
        if (galleryAlbumsPerPage() != 0) {
            return (int) ceil($_zp_gallery->getNumAlbums() / galleryAlbumsPerPage());
        } else {
            return NULL;
        }
        return NULL;
    } else if (isset($_zp_zenpage)) {
        if (in_context(ZP_ZENPAGE_NEWS_CATEGORY)) {
            $cat = $_zp_current_category;
        } else {
            $cat = NULL;
        }
        return (int) ceil(count($_zp_zenpage->getArticles(0, NULL, true, NULL, NULL, NULL, $cat)) / ZP_ARTICLES_PER_PAGE);
    }
}

/**
 * Returns the URL of the page number passed as a parameter
 *
 * @param int $page Which page is desired
 * @param int $total How many pages there are.
 * @return int
 */
function getPageNumURL($page, $total = null) {
    global $_zp_current_album, $_zp_gallery, $_zp_current_search, $_zp_gallery_page, $_zp_conf_vars;
    if (is_null($total)) {
        $total = getTotalPages();
    }
    if ($page <= 0 || $page > $total) {
        return NULL;
    }
    if (in_context(ZP_SEARCH)) {
        $searchwords = $_zp_current_search->codifySearchString();
        $searchdate = $_zp_current_search->getSearchDate();
        $searchfields = $_zp_current_search->getSearchFields(true);
        $searchpagepath = getSearchURL($searchwords, $searchdate, $searchfields, $page, array('albums' => $_zp_current_search->getAlbumList()));
        return $searchpagepath;
    } else if (in_context(ZP_ALBUM)) {
        return $_zp_current_album->getLink($page);
    } else if (in_array($_zp_gallery_page, array('index.php', 'album.php', 'image.php'))) {
        if (in_context(ZP_INDEX)) {
            $pagination1 = '/';
            $pagination2 = 'index.php';
            if ($page > 1) {
                $pagination1 .= _PAGE_ . '/' . $page . '/';
                $pagination2 .= '?page=' . $page;
            }
        } else {
            return NULL;
        }
    } else {
        // handle custom page
        $pg = stripSuffix($_zp_gallery_page);
        if (array_key_exists($pg, $_zp_conf_vars['special_pages'])) {
            $pagination1 = preg_replace('~^_PAGE_/~', _PAGE_ . '/', $_zp_conf_vars['special_pages'][$pg]['rewrite']) . '/';
        } else {
            $pagination1 = '/' . _PAGE_ . '/' . $pg . '/';
        }
        $pagination2 = 'index.php?p=' . $pg;
        if ($page > 1) {
            $pagination1 .= $page . '/';
            $pagination2 .= '&page=' . $page;
        }
    }
    return zp_apply_filter('getLink', rewrite_path($pagination1, $pagination2), $_zp_gallery_page, $page);
}

/**
 * Returns true if there is a next page
 *
 * @return bool
 */
function hasNextPage() {
    return (getCurrentPage() < getTotalPages());
}

/**
 * Returns the URL of the next page. Use within If or while loops for pagination.
 *
 * @return string
 */
function getNextPageURL() {
    return getPageNumURL(getCurrentPage() + 1);
}

/**
 * Prints the URL of the next page.
 *
 * @param string $text text for the URL
 * @param string $title Text for the HTML title
 * @param string $class Text for the HTML class
 * @param string $id Text for the HTML id
 */
function printNextPageURL($text, $title = NULL, $class = NULL, $id = NULL) {
    if (hasNextPage()) {
        printLinkHTML(getNextPageURL(), $text, $title, $class, $id);
    } else {
        echo "<span class=\"disabledlink\">$text</span>";
    }
}

/**
 * Returns TRUE if there is a previous page. Use within If or while loops for pagination.
 *
 * @return bool
 */
function hasPrevPage() {
    return (getCurrentPage() > 1);
}

/**
 * Returns the URL of the previous page.
 *
 * @return string
 */
function getPrevPageURL() {
    return getPageNumURL(getCurrentPage() - 1);
}

/**
 * Returns the URL of the previous page.
 *
 * @param string $text The linktext that should be printed as a link
 * @param string $title The text the html-tag "title" should contain
 * @param string $class Insert here the CSS-class name you want to style the link with
 * @param string $id Insert here the CSS-ID name you want to style the link with
 */
function printPrevPageURL($text, $title = NULL, $class = NULL, $id = NULL) {
    if (hasPrevPage()) {
        printLinkHTML(getPrevPageURL(), $text, $title, $class, $id);
    } else {
        echo "<span class=\"disabledlink\">$text</span>";
    }
}

/**
 * Prints a page navigation including previous and next page links
 *
 * @param string $prevtext Insert here the linktext like 'previous page'
 * @param string $separator Insert here what you like to be shown between the prev and next links
 * @param string $nexttext Insert here the linktext like "next page"
 * @param string $class Insert here the CSS-class name you want to style the link with (default is "pagelist")
 * @param string $id Insert here the CSS-ID name if you want to style the link with this
 */
function printPageNav($prevtext, $separator, $nexttext, $class = 'pagenav', $id = NULL) {
    echo "<div" . (($id) ? " id=\"$id\"" : "") . " class=\"$class\">";
    printPrevPageURL($prevtext, gettext("Previous Page"));
    echo " $separator ";
    printNextPageURL($nexttext, gettext("Next Page"));
    echo "</div>\n";
}

/**
 * Prints a list of all pages.
 *
 * @param string $class the css class to use, "pagelist" by default
 * @param string $id the css id to use
 * @param int $navlen Number of navigation links to show (0 for all pages). Works best if the number is odd.
 */
function printPageList($class = 'pagelist', $id = NULL, $navlen = 9) {
    printPageListWithNav(null, null, false, false, $class, $id, false, $navlen);
}

/**
 * returns a page nav list.
 *
 * @param bool $_oneImagePage set to true if there is only one image page as, for instance, in flash themes
 * @param int $navlen Number of navigation links to show (0 for all pages). Works best if the number is odd.
 * @param bool $firstlast Add links to the first and last pages of you gallery
 * @param int $current the current page
 * @param int $total total number of pages
 *
 */
function getPageNavList($_oneImagePage, $navlen, $firstlast, $current, $total) {
    $result = array();
    if (hasPrevPage()) {
        $result['prev'] = getPrevPageURL();
    } else {
        $result['prev'] = NULL;
    }
    if ($firstlast) {
        $result[1] = getPageNumURL(1, $total);
    }

    if ($navlen == 0) {
        $navlen = $total;
    }
    $extralinks = 2;
    if ($firstlast)
        $extralinks = $extralinks + 2;
    $len = floor(($navlen - $extralinks) / 2);
    $j = max(round($extralinks / 2), min($current - $len - (2 - round($extralinks / 2)), $total - $navlen + $extralinks - 1));
    $ilim = min($total, max($navlen - round($extralinks / 2), $current + floor($len)));
    $k1 = round(($j - 2) / 2) + 1;
    $k2 = $total - round(($total - $ilim) / 2);

    for ($i = $j; $i <= $ilim; $i++) {
        $result[$i] = getPageNumURL($i, $total);
    }
    if ($firstlast) {
        $result[$total] = getPageNumURL($total, $total);
    }
    if (hasNextPage()) {
        $result['next'] = getNextPageURL();
    } else {
        $result['next'] = NULL;
    }
    return $result;
}

/**
 * Prints a full page navigation including previous and next page links with a list of all pages in between.
 *
 * @param string $prevtext Insert here the linktext like 'previous page'
 * @param string $nexttext Insert here the linktext like 'next page'
 * @param bool $_oneImagePage set to true if there is only one image page as, for instance, in flash themes
 * @param string $nextprev set to true to get the 'next' and 'prev' links printed
 * @param string $class Insert here the CSS-class name you want to style the link with (default is "pagelist")
 * @param string $id Insert here the CSS-ID name if you want to style the link with this
 * @param bool $firstlast Add links to the first and last pages of you gallery
 * @param int $navlen Number of navigation links to show (0 for all pages). Works best if the number is odd.
 */
function printPageListWithNav($prevtext, $nexttext, $_oneImagePage = false, $nextprev = true, $class = 'pagelist', $id = NULL, $firstlast = true, $navlen = 9) {
    $current = getCurrentPage();
    $total = max(1, getTotalPages($_oneImagePage));
    $nav = getPageNavList($_oneImagePage, $navlen, $firstlast, $current, $total);
    if ($total > 1) {
        ?>
        <div <?php if ($id) echo ' id="'.$id.'"'; ?> class="<?php echo $class; ?>">
            <ul class="<?php echo $class; ?>">
                <?php
                $prev = $nav['prev'];
                unset($nav['prev']);
                $next = $nav['next'];
                unset($nav['next']);
                if ($nextprev) {
                    ?>
                    <li class="prev">
                        <?php
                        if ($prev) {
                            printLinkHTML($prev, html_encode($prevtext), gettext('Previous Page'));
                        } else {
                            ?>
                            <span class="disabledlink"><?php echo html_encode($prevtext); ?></span>
                            <?php
                        }
                        ?>
                    </li>
                    <?php
                }
                $last = NULL;
                if ($firstlast) {
                    ?>
                    <li class="<?php
                    if ($current == 1)
                        echo 'current';
                    else
                        echo 'first';
                    ?>">
                                <?php
                                if ($current == 1) {
                                    echo '1';
                                } else {
                                    printLinkHTML($nav[1], 1, gettext("Page 1"));
                                }
                                ?>
                    </li>
                    <?php
                    $last = 1;
                    unset($nav[1]);
                }
                foreach ($nav as $i => $link) {
                    $d = $i - $last;
                    if ($d > 2) {
                        ?>
                        <li>
                            <?php
                            $k1 = $i - (int) (($i - $last) / 2);
                            printLinkHTML(getPageNumURL($k1, $total), '...', sprintf(ngettext('Page %u', 'Page %u', $k1), $k1));
                            ?>
                        </li>
                        <?php
                    } else if ($d == 2) {
                        ?>
                        <li>
                            <?php
                            $k1 = $last + 1;
                            printLinkHTML(getPageNumURL($k1, $total), $k1, sprintf(ngettext('Page %u', 'Page %u', $k1), $k1));
                            ?>
                        </li>
                        <?php
                    }
                    ?>
                    <li<?php if ($current == $i) echo ' class="current"'; ?>>
                        <?php
                        if ($i == $current) {
                            echo $i;
                        } else {
                            $title = sprintf(ngettext('Page %1$u', 'Page %1$u', $i), $i);
                            printLinkHTML($link, $i, $title);
                        }
                        ?>
                    </li>
                    <?php
                    $last = $i;
                    unset($nav[$i]);
                    if ($firstlast && count($nav) == 1) {
                        break;
                    }
                }
                if ($firstlast) {
                    foreach ($nav as $i => $link) {
                        $d = $i - $last;
                        if ($d > 2) {
                            $k1 = $i - (int) (($i - $last) / 2);
                            ?>
                            <li>
                                <?php printLinkHTML(getPageNumURL($k1, $total), '...', sprintf(ngettext('Page %u', 'Page %u', $k1), $k1)); ?>
                            </li>
                            <?php
                        } else if ($d == 2) {
                            $k1 = $last + 1;
                            ?>
                            <li>
                                <?php printLinkHTML(getPageNumURL($k1, $total), $k1, sprintf(ngettext('Page %u', 'Page %u', $k1), $k1)); ?>
                            </li>
                            <?php
                        }
                        ?>
                        <li class="last<?php if ($current == $i) echo ' current'; ?>">
                            <?php
                            if ($current == $i) {
                                echo $i;
                            } else {
                                printLinkHTML($link, $i, sprintf(ngettext('Page %u', 'Page %u', $i), $i));
                            }
                            ?>
                        </li>
                        <?php
                    }
                }
                if ($nextprev) {
                    ?>
                    <li class="next">
                        <?php
                        if ($next) {
                            printLinkHTML($next, html_encode($nexttext), gettext('Next Page'));
                        } else {
                            ?>
                            <span class="disabledlink"><?php echo html_encode($nexttext); ?></span>
                            <?php
                        }
                        ?>
                    </li>
                    <?php
                }
                ?>
            </ul>
        </div>
        <?php
    }
}

//*** Album Context ************************
//******************************************

/**
 * Sets the album passed as the current album
 *
 * @param object $album the album to be made current
 */
function makeAlbumCurrent($album) {
    global $_zp_current_album;
    $_zp_current_album = $album;
    set_context(ZP_INDEX | ZP_ALBUM);
}

/**
 * Returns the raw title of the current album.
 *
 * @return string
 */
function getAlbumTitle() {
    if (!in_context(ZP_ALBUM))
        return false;
    global $_zp_current_album;
    return $_zp_current_album->getTitle();
}

/**
 * Returns a text-only title of the current album.
 *
 * @return string
 */
function getBareAlbumTitle() {
    return getBare(getAlbumTitle());
}

/**
 * Returns an album title taged with of Not visible or password protected status
 *
 * @return string;
 */
function getAnnotatedAlbumTitle() {
    global $_zp_current_album;
    $title = getBareAlbumTitle();
    $pwd = $_zp_current_album->getPassword();
    if (zp_loggedin() && !empty($pwd)) {
        $title .= "\n" . gettext('The album is password protected.');
    }
    if (!$_zp_current_album->isPublished()) {
        $title .= "\n" . gettext('The album is un-published.');
    }
    return $title;
}

function printAnnotatedAlbumTitle() {
    echo html_encode(getAnnotatedAlbumTitle());
}

/**
 * Prints an encapsulated title of the current album.
 * If you are logged in you can click on this to modify the title on the fly.
 *
 * @author Ozh
 */
function printAlbumTitle() {
    echo html_encodeTagged(getAlbumTitle());
}

function printBareAlbumTitle() {
    echo html_encodeTagged(getBareAlbumTitle());
}

/**
 * Gets the 'n' for n of m albums
 *
 * @return int
 */
function albumNumber() {
    global $_zp_current_album, $_zp_current_image, $_zp_current_search, $_zp_gallery;
    $name = $_zp_current_album->getFileName();
    if (in_context(ZP_SEARCH)) {
        $albums = $_zp_current_search->getAlbums();
    } else if (in_context(ZP_ALBUM)) {
        $parent = $_zp_current_album->getParent();
        if (is_null($parent)) {
            $albums = $_zp_gallery->getAlbums();
        } else {
            $albums = $parent->getAlbums();
        }
    }
    $c = 0;
    foreach ($albums as $albumfolder) {
        $c++;
        if ($name == $albumfolder) {
            return $c;
        }
    }
    return false;
}

/**
 * Returns an array of the names of the parents of the current album.
 *
 * @param object $album optional album object to use inseted of the current album
 * @return array
 */
function getParentAlbums($album = null) {
    $parents = array();
    if (in_context(ZP_ALBUM)) {
        global $_zp_current_album, $_zp_current_search, $_zp_gallery;
        if (is_null($album)) {
            if (in_context(ZP_SEARCH_LINKED) && !in_context(ZP_ALBUM_LINKED)) {
                $album = $_zp_current_search->getDynamicAlbum();
                if (empty($album))
                    return $parents;
            } else {
                $album = $_zp_current_album;
            }
        }
        while (!is_null($album = $album->getParent())) {
            array_unshift($parents, $album);
        }
    }
    return $parents;
}

/**
 * returns the breadcrumb item for the current images's album
 *
 * @param string $title Text to be used as the URL title tag
 * @return array
 */
function getAlbumBreadcrumb($title = NULL) {
    global $_zp_current_search, $_zp_gallery, $_zp_current_album, $_zp_last_album;
    $output = array();
    if (in_context(ZP_SEARCH_LINKED)) {
        $album = NULL;
        $dynamic_album = $_zp_current_search->getDynamicAlbum();
        if (empty($dynamic_album)) {
            if (!is_null($_zp_current_album)) {
                if (in_context(ZP_ALBUM_LINKED) && $_zp_last_album == $_zp_current_album->name) {
                    $album = $_zp_current_album;
                }
            }
        } else {
            if (in_context(ZP_IMAGE) && in_context(ZP_ALBUM_LINKED)) {
                $album = $_zp_current_album;
            } else {
                $album = $dynamic_album;
            }
        }
    } else {
        $album = $_zp_current_album;
    }
    if ($album) {
        if (is_null($title)) {
            $title = $album->getTitle();
            if (empty($title)) {
                $title = gettext('Album Thumbnails');
            }
        }
        return array('link' => $album->getLink(), 'text' => $title, 'title' => getBare($title));
    }
    return false;
}

/**
 * prints the breadcrumb item for the current images's album
 *
 * @param string $before Text to place before the breadcrumb
 * @param string $after Text to place after the breadcrumb
 * @param string $title Text to be used as the URL title attribute and text link
 */
function printAlbumBreadcrumb($before = '', $after = '', $title = NULL) {
    if ($breadcrumb = getAlbumBreadcrumb($title)) {
        if ($before) {
            $output = '<span class="beforetext">' . html_encode($before) . '</span>';
        } else {
            $output = '';
        }
        $output .= '<a href="' . html_encode($breadcrumb['link']) . '" title="' . html_encode($breadcrumb['title']) . '">';
        $output .= html_encode($breadcrumb['text']);
        $output .= '</a>';
        if ($after) {
            $output .= '<span class="aftertext">' . html_encode($after) . '</span>';
        }
        echo $output;
    }
}

/**
 * Prints the "breadcrumb" for a search page
 *      if the search was for a data range, the breadcrumb is "Archive"
 *      otherwise it is "Search"
 * @param string $between Insert here the text to be printed between the links
 * @param string $class is the class for the link (if present)
 * @param string $search text for a search page title
 * @param string $archive text for an archive page title
 * @param string $format data format for archive page crumb
 */
function printSearchBreadcrumb($between = NULL, $class = NULL, $search = NULL, $archive = NULL, $format = '%B %Y') {
    global $_zp_current_search;
    if (is_null($between)) {
        $between = ' | ';
    }
    if ($class) {
        $class = ' class="' . $class . '"';
    }
    if ($d = $_zp_current_search->getSearchDate()) {
        if (is_null($archive)) {
            $text = gettext('Archive');
            $textdecoration = true;
        } else {
            $text = getBare(html_encode($archive));
            $textdecoration = false;
        }
        echo "<a href=\"" . html_encode(getCustomPageURL('archive', NULL)) . "\"$class title=\"" . $text . "\">";
        printf('%s' . $text . '%s', $textdecoration ? '<em>' : '', $textdecoration ? '</em>' : '');
        echo "</a>";
        echo '<span class="betweentext">' . html_encode($between) . '</span>';
        if ($format) {
            $d = strtotime($d);
            $d = strftime($format, $d);
        }
        echo $d;
    } else {
        if (is_null($search)) {
            $text = gettext('Search');
            $textdecoration = true;
        } else {
            $text = getBare(html_encode($search));
            $textdecoration = false;
        }
        printf('%s' . $text . '%s', $textdecoration ? '<em>' : '', $textdecoration ? '</em>' : '');
    }
}

/**
 * returns the breadcrumb navigation for album, gallery and image view.
 *
 * @return array
 */
function getParentBreadcrumb() {
    global $_zp_gallery, $_zp_current_search, $_zp_current_album, $_zp_last_album;
    $output = array();
    if (in_context(ZP_SEARCH_LINKED)) {
        $page = $_zp_current_search->page;
        $searchwords = $_zp_current_search->getSearchWords();
        $searchdate = $_zp_current_search->getSearchDate();
        $searchfields = $_zp_current_search->getSearchFields(true);
        $search_album_list = $_zp_current_search->getAlbumList();
        if (!is_array($search_album_list)) {
            $search_album_list = array();
        }
        $searchpagepath = getSearchURL($searchwords, $searchdate, $searchfields, $page, array('albums' => $search_album_list));
        $dynamic_album = $_zp_current_search->getDynamicAlbum();
        if (empty($dynamic_album)) {
            if (empty($searchdate)) {
                $output[] = array('link' => $searchpagepath, 'title' => gettext("Return to search"), 'text' => gettext("Search"));
                if (is_null($_zp_current_album)) {
                    return $output;
                } else {
                    $parents = getParentAlbums();
                }
            } else {
                return array(array('link' => $searchpagepath, 'title' => gettext("Return to archive"), 'text' => gettext("Archive")));
            }
        } else {
            $album = $dynamic_album;
            $parents = getParentAlbums($album);
            if (in_context(ZP_ALBUM_LINKED)) {
                array_push($parents, $album);
            }
        }
// remove parent links that are not in the search path
        foreach ($parents as $key => $analbum) {
            $target = $analbum->name;
            if ($target !== $dynamic_album && !in_array($target, $search_album_list)) {
                unset($parents[$key]);
            }
        }
    } else {
        $parents = getParentAlbums();
    }
    $n = count($parents);
    if ($n > 0) {
        array_push($parents, $_zp_current_album);
        $index = -1;
        foreach ($parents as $parent) {
            $index++;
            if($index != 0) {
                $parentparent = $parents[$index-1];
                $page = $parent->getGalleryPage();
                $url = $parentparent->getLink($page);
                $output[] = array('link' => html_encode($url), 'title' => $parentparent->getTitle(), 'text' => $parentparent->getTitle());
            }
        }
    }
    return $output;
}

/**
 * Prints the breadcrumb navigation for album, gallery and image view.
 *
 * @param string $before Insert here the text to be printed before the links
 * @param string $between Insert here the text to be printed between the links
 * @param string $after Insert here the text to be printed after the links
 * @param mixed $truncate if not empty, the max lenght of the description.
 * @param string $elipsis the text to append to the truncated description
 */
function printParentBreadcrumb($before = NULL, $between = NULL, $after = NULL, $truncate = NULL, $elipsis = NULL) {
    $crumbs = getParentBreadcrumb();
    if (!empty($crumbs)) {
        if (is_null($between)) {
            $between = ' | ';
        }
        if (is_null($after)) {
            $after = ' | ';
        }
        if (is_null($elipsis)) {
            $elipsis = '...';
        }
        if ($before) {
            $output = '<span class="beforetext">' . html_encode($before) . '</span>';
        } else {
            $output = '';
        }
        if ($between) {
            $between = '<span class="betweentext">' . html_encode($between) . '</span>';
        }
        $i = 0;
        foreach ($crumbs as $crumb) {
            if ($i > 0) {
                $output .= $between;
            }
//cleanup things in description for use as attribute tag
            $desc = $crumb['title'];
            if (!empty($desc) && $truncate) {
                $desc = truncate_string($desc, $truncate, $elipsis);
            }
            $output .= '<a href="' . html_encode($crumb['link']) . '"' . ' title="' . html_encode(getBare($desc)) . '">' . html_encode($crumb['text']) . '</a>';
            $i++;
        }
        if ($after) {
            $output .= '<span class="aftertext">' . html_encode($after) . '</span>';
        }
        echo $output;
    }
}

/**
 * Prints a link to the 'main website', not the Zenphoto site home page!
 * Only prints the link if the url is not empty and does not point back the gallery page
 *
 * @param string $before text to precede the link
 * @param string $after text to follow the link
 * @param string $title Title text
 * @param string $class optional css class
 * @param string $id optional css id
 *  */
function printHomeLink($before = '', $after = '', $title = NULL, $class = NULL, $id = NULL) {
    global $_zp_gallery;
    $site = rtrim($_zp_gallery->getWebsiteURL(), '/');
    if (!empty($site)) {
        $name = $_zp_gallery->getWebsiteTitle();
        if (empty($name)) {
            $name = gettext('Home');
        }
        if ($site != SEO_FULLWEBPATH) {
            if ($before) {
                echo '<span class="beforetext">' . html_encode($before) . '</span>';
            }
            printLinkHTML($site, $name, $title, $class, $id);
            if ($after) {
                echo '<span class="aftertext">' . html_encode($after) . '</span>';
            }
        }
    }
}

/**
 * Returns the formatted date field of the album
 *
 * @param string $format optional format string for the date
 * @return string
 */
function getAlbumDate($format = null) {
    global $_zp_current_album;
    $d = $_zp_current_album->getDateTime();
    if (empty($d) || ($d == '0000-00-00 00:00:00')) {
        return false;
    }
    if (is_null($format)) {
        return $d;
    }
    return zpFormattedDate($format, strtotime($d));
}

/**
 * Prints the date of the current album
 *
 * @param string $before Insert here the text to be printed before the date.
 * @param string $format Format string for the date formatting
 */
function printAlbumDate($before = '', $format = NULL) {
    global $_zp_current_album;
    if (is_null($format)) {
        $format = DATE_FORMAT;
    }
    $date = getAlbumDate($format);
    if ($date) {
        if ($before) {
            $date = '<span class="beforetext">' . $before . '</span>' . $date;
        }
    }
    echo html_encodeTagged($date);
}

/**
 * Returns the Location of the album.
 *
 * @return string
 */
function getAlbumLocation() {
    global $_zp_current_album;
    return $_zp_current_album->getLocation();
}

/**
 * Prints the location of the album
 *
 * @author Ozh
 */
function printAlbumLocation() {
    echo html_encodeTagged(getAlbumLocation());
}

/**
 * Returns the raw description of the current album.
 *
 * @return string
 */
function getAlbumDesc() {
    if (!in_context(ZP_ALBUM))
        return false;
    global $_zp_current_album;
    return $_zp_current_album->getDesc();
}

/**
 * Returns a text-only description of the current album.
 *
 * @return string
 */
function getBareAlbumDesc() {
    return getBare(getAlbumDesc());
}

/**
 * Prints description of the current album
 *
 * @author Ozh
 */
function printAlbumDesc() {
    global $_zp_current_album;
    echo html_encodeTagged(getAlbumDesc());
}

function printBareAlbumDesc() {
    echo html_encode(getBareAlbumDesc());
}

/**
 * Returns the custom_data field of the current album
 *
 * @return string
 */
function getAlbumCustomData() {
    global $_zp_current_album;
    return $_zp_current_album->getCustomData();
}

/**
 * Prints the custom_data field of the current album.
 * Converts and displays line break in the admin field as <br />.
 *
 * @author Ozh
 */
function printAlbumCustomData() {
    echo html_encodeTagged(getAlbumCustomData());
}

/**
 * A composit for getting album data
 *
 * @param string $field which field you want
 * @return string
 */
function getAlbumData($field) {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_album_image;
    return get_language_string($_zp_album_image->get($field));
}

/**
 * Prints arbitrary data from the album object
 *
 * @param string $field the field name of the data desired
 * @param string $label text to label the field
 * @author Ozh
 */
function printAlbumData($field, $label = '') {
    global $_zp_current_album;
    echo html_encodeTagged($_zp_current_album->get($field));
}

/**
 * Returns the album page number of the current image
 *
 * @param object $album optional album object
 * @return integer
 */
function getAlbumPage($album = NULL) {
    global $_zp_current_album, $_zp_current_image, $_zp_current_search, $_firstPageImages;
    if (is_null($album))
        $album = $_zp_current_album;
    $page = 0;
    if (in_context(ZP_IMAGE) && !in_context(ZP_SEARCH)) {
        $imageindex = $_zp_current_image->getIndex();
        $numalbums = $album->getNumAlbums();
        $imagepage = floor(($imageindex - $_firstPageImages) / max(1, getOption('images_per_page'))) + 1;
        $albumpages = ceil($numalbums / max(1, getOption('albums_per_page')));
        if ($albumpages == 0 && $_firstPageImages > 0)
            $imagepage++;
        $page = $albumpages + $imagepage;
    }
    return $page;
}

/**
 * Returns the album link url of the current album.
 *
 * @param object $album optional album object
 * @return string
 */
function getAlbumURL($album = NULL) {
    global $_zp_current_album;
    if (is_null($album))
        $album = $_zp_current_album;
    if (in_context(ZP_IMAGE)) {
        $page = getAlbumPage($album);
        if ($page <= 1)
            $page = 0;
    } else {
        $page = 0;
    }
    return $album->getLink($page);
}

/**
 * Prints the album link url of the current album.
 *
 * @param string $text Insert the link text here.
 * @param string $title Insert the title text here.
 * @param string $class Insert here the CSS-class name with with you want to style the link.
 * @param string $id Insert here the CSS-id name with with you want to style the link.
 */
function printAlbumURL($text, $title, $class = NULL, $id = NULL) {
    printLinkHTML(getAlbumURL(), $text, $title, $class, $id);
}

/**
 * Returns the name of the defined album thumbnail image.
 *
 * @return string
 */
function getAlbumThumb() {
    global $_zp_current_album;
    return $_zp_current_album->getThumb();
}

/**
 * Returns an img src link to the password protect thumb substitute
 *
 * @param string $extra extra stuff to put in the HTML
 * @return string
 */
function getPasswordProtectImage($extra = '') {
    global $_zp_themeroot;
    $image = '';
    $themedir = SERVERPATH . '/themes/' . basename($_zp_themeroot);
    if (file_exists(internalToFilesystem($themedir . '/images/err-passwordprotected.png'))) {
        $image = $_zp_themeroot . '/images/err-passwordprotected.png';
    } else if (file_exists(internalToFilesystem($themedir . '/images/err-passwordprotected.gif'))) {
        $image = $_zp_themeroot . '/images/err-passwordprotected.gif';
    } else {
        $image = WEBPATH . '/' . ZENFOLDER . '/images/err-passwordprotected.png';
    }
    return '<img src="' . $image . '" ' . $extra . ' alt="protected" loading="lazy" />';
}

/**
 * Prints the album thumbnail image.
 *
 * @param string $alt Insert the text for the alternate image name here.
 * @param string $class Insert here the CSS-class name with with you want to style the link.
 * @param string $id Insert here the CSS-id name with with you want to style the link.
 * @param string $title option title attribute
 *  */
function printAlbumThumbImage($alt, $class = NULL, $id = NULL , $title = null) {
    global $_zp_current_album;
    $thumbobj = $_zp_current_album->getAlbumThumbImage();
    $sizes = getSizeDefaultThumb($thumbobj);
    if (empty($title)) {
        $title = $alt;
    }
    $attr = array(
            'src' => html_pathurlencode($thumbobj->getThumb('album')),
            'alt' => html_encode($alt),
            'title' => html_encode($title),
            'class' => $class,
            'id' => $id,
            'width' => $sizes[0],
            'height' => $sizes[1],
            'loading' => 'lazy'
    );
    if (!$_zp_current_album->isPublished()) {
        $attr['class'] .= " not_visible";
    }
    $pwd = $_zp_current_album->getPassword();
    if (!empty($pwd)) {
        $attr['class'] .= " password_protected";
    }
    $attr['class'] = trim($attr['class']);
    $attr_filtered = zp_apply_filter('standard_album_thumb_attr', $attr, $thumbobj);
    if (!getOption('use_lock_image') || $_zp_current_album->isMyItem(LIST_RIGHTS) || empty($pwd)) {
        $attributes = generateAttributesFromArray($attr_filtered);
        $html = '<img' . $attributes . ' />';
        $html = zp_apply_filter('standard_album_thumb_html', $html, $thumbobj);
        echo $html;
    } else {
        $size = ' width="' . $attr['width'] . '"';
        echo getPasswordProtectImage($size);
    }
}

/**
 * Returns a link to a custom sized thumbnail of the current album
 *
 * @param int $size the size of the image to have
 * @param int $width width
 * @param int $height height
 * @param int $cropw crop width
 * @param int $croph crop height
 * @param int $cropx crop part x axis
 * @param int $cropy crop part y axis
 * @param bool $effects image effects (e.g. set 'gray' to force grayscale)
 *
 * @return string
 */
function getCustomAlbumThumb($size, $width = NULL, $height = NULL, $cropw = NULL, $croph = NULL, $cropx = NULL, $cropy = null, $effects = NULL) {
    global $_zp_current_album;
    $thumb = $_zp_current_album->getAlbumThumbImage();
    return $thumb->getCustomImage($size, $width, $height, $cropw, $croph, $cropx, $cropy, true, $effects);
}

/**
 * Prints a link to a custom sized thumbnail of the current album
 *
 * See getCustomImageURL() for details.
 *
 * @param string $alt Alt atribute text
 * @param int $size size
 * @param int $width width
 * @param int $height height
 * @param int $cropw cropwidth
 * @param int $croph crop height
 * @param int $cropx crop part x axis
 * @param int $cropy crop part y axis
 * @param string $class css class
 * @param string $id css id
 * @param string $title title attribute
 * @param bool $maxspace true for maxspace image, false is default
 *
 * @return string
 */
function printCustomAlbumThumbImage($alt, $size, $width = NULL, $height = NULL, $cropw = NULL, $croph = NULL, $cropx = NULL, $cropy = null, $class = NULL, $id = NULL, $title = null, $maxspace = false) {
    global $_zp_current_album;
    $thumbobj = $_zp_current_album->getAlbumThumbImage();
    $sizes = getSizeCustomImage($size, $width, $height, $cropw, $croph, $cropx, $cropy, $thumbobj, 'thumb');
    if (empty($title)) {
        $title = $alt;
    }
    $attr = array(
            'alt' => html_encode($alt),
            'class' => $class,
            'title' => html_encode($title),
            'id' => $id,
            'width' => $sizes[0],
            'height' => $sizes[1],
            'loading' => 'lazy'
    );
    if($maxspace) {
        getMaxSpaceContainer($width, $height, $thumbobj, true);
    }
    if (!$_zp_current_album->isPublished()) {
        $attr['class'] .= " not_visible";
    }
    $pwd = $_zp_current_album->getPassword();
    if (!empty($pwd)) {
        $attr['class'] .= " password_protected";
    }
    $attr['class'] = trim($attr['class']);
    if ($maxspace) {
        $attr['src']= html_pathurlencode(getCustomAlbumThumb(null, $width, $height, null, null, null, null));
    } else {
        $attr['src']= html_pathurlencode(getCustomAlbumThumb($size, $width, $height, $cropw, $croph, $cropx, $cropy));
    }
    $attr_filtered = zp_apply_filter('custom_album_thumb_attr', $attr, $thumbobj);
    if (!getOption('use_lock_image') || $_zp_current_album->isMyItem(LIST_RIGHTS) || empty($pwd)) {
        $attributes = generateAttributesFromArray($attr_filtered);
        $html = '<img' . $attributes . ' />';
        $html = zp_apply_filter('custom_album_thumb_html', $html, $thumbobj);
        echo $html;
    } else {
        $size = ' width="' . $attr['width'] . '"';
        echo getPasswordProtectImage($size);
    }
}

/**
 * Called by ***MaxSpace functions to compute the parameters to be passed to xxCustomyyy functions.
 *
 * @param int $width maxspace width
 * @param int $height maxspace height
 * @param object $image the image in question
 * @param bool $thumb true if for a thumbnail
 */
function getMaxSpaceContainer(&$width, &$height, $image, $thumb = false) {
    global $_zp_gallery;
    $upscale = getOption('image_allow_upscale');
    $imagename = $image->filename;
    if ($thumb) {
        $s_width = $image->getThumbWidth();
        $s_height = $image->getThumbHeight();
    } else {
        $s_width = $image->get('width');
        if ($s_width == 0)
            $s_width = max($width, $height);
        $s_height = $image->get('height');
        if ($s_height == 0)
            $s_height = max($width, $height);
    }

    $newW = round($height / $s_height * $s_width);
    $newH = round($width / $s_width * $s_height);
    if (DEBUG_IMAGE)
        debugLog("getMaxSpaceContainer($width, $height, $imagename, $thumb): \$s_width=$s_width; \$s_height=$s_height; \$newW=$newW; \$newH=$newH; \$upscale=$upscale;");
    if ($newW > $width) {
        if ($upscale || $s_height > $newH) {
            $height = $newH;
        } else {
            $height = $s_height;
            $width = $s_width;
        }
    } else {
        if ($upscale || $s_width > $newW) {
            $width = $newW;
        } else {
            $height = $s_height;
            $width = $s_width;
        }
    }
}

/**
 * Returns a link to a un-cropped custom sized version of the current album thumb within the given height and width dimensions.
 *
 * @param int $width width
 * @param int $height height
 * @return string
 */
function getCustomAlbumThumbMaxSpace($width, $height) {
    global $_zp_current_album;
    $albumthumb = $_zp_current_album->getAlbumThumbImage();
    getMaxSpaceContainer($width, $height, $albumthumb, true);
    return getCustomAlbumThumb(NULL, $width, $height, NULL, NULL, NULL, NULL);
}

/**
 * Prints a un-cropped custom sized album thumb within the given height and width dimensions.
 * Note: a class of 'not_visible' or 'password_protected' will be added as appropriate
 *
 * @param string $alt Alt text for the url
 * @param int $width width
 * @param int $height height
 * @param string $class Optional style class
 * @param string $id Optional style id
 * @param string $title Optional title attribute
 */
function printCustomAlbumThumbMaxSpace($alt, $width, $height, $class = NULL, $id = NULL, $title = null) {
    printCustomAlbumThumbImage($alt, NULL, $width, $height, NULL, NULL, NULL, NULL, $class, $id, $title, true);
}

/**
 * Returns the next album
 *
 * @return object
 */
function getNextAlbum() {
    global $_zp_current_album, $_zp_current_search, $_zp_gallery;
    if (in_context(ZP_SEARCH) || in_context(ZP_SEARCH_LINKED)) {
        $nextalbum = $_zp_current_search->getNextAlbum($_zp_current_album->name);
    } else if (in_context(ZP_ALBUM)) {
        $nextalbum = $_zp_current_album->getNextAlbum();
    } else {
        return null;
    }
    return $nextalbum;
}

/**
 * Get the URL of the next album in the gallery.
 *
 * @return string
 */
function getNextAlbumURL() {
    $nextalbum = getNextAlbum();
    if ($nextalbum) {
        return $nextalbum->getLink();
    }
    return false;
}

/**
 * Returns the previous album
 *
 * @return object
 */
function getPrevAlbum() {
    global $_zp_current_album, $_zp_current_search;
    if (in_context(ZP_SEARCH) || in_context(ZP_SEARCH_LINKED)) {
        $prevalbum = $_zp_current_search->getPrevAlbum($_zp_current_album->name);
    } else if (in_context(ZP_ALBUM)) {
        $prevalbum = $_zp_current_album->getPrevAlbum();
    } else {
        return null;
    }
    return $prevalbum;
}

/**
 * Get the URL of the previous album in the gallery.
 *
 * @return string
 */
function getPrevAlbumURL() {
    $prevalbum = getPrevAlbum();
    if ($prevalbum) {
        return $prevalbum->getLink();
    }
    return false;
}

/**
 * Returns true if this page has image thumbs on it
 *
 * @return bool
 */
function isImagePage() {
    if (getNumImages()) {
        global $_zp_page, $_firstPageImages;
        $imagestart = getTotalPages(2); // # of album pages
        if (!$_firstPageImages)
            $imagestart++; // then images start on the last album page.
        return $_zp_page >= $imagestart;
    }
    return false;
}

/**
 * Returns true if this page has album thumbs on it
 *
 * @return bool
 */
function isAlbumPage() {
    global $_zp_page;
    $pageCount = Ceil(getNumAlbums() / max(1, getOption('albums_per_page')));
    return ($_zp_page <= $pageCount);
}

/**
 * Returns the number of images in the album.
 *
 * @return int
 */
function getNumImages() {
    global $_zp_current_album, $_zp_current_search;
    if ((in_context(ZP_SEARCH_LINKED) && !in_context(ZP_ALBUM_LINKED)) || in_context(ZP_SEARCH) && is_null($_zp_current_album)) {
        return $_zp_current_search->getNumImages();
    } else {
        return $_zp_current_album->getNumImages();
    }
}

/**
 * Returns the next image on a page.
 * sets $_zp_current_image to the next image in the album.

 * Returns true if there is an image to be shown
 *
 * @param bool $all set to true disable pagination
 * @param int $firstPageCount the number of images which can go on the page that transitions between albums and images
 *                          Normally this parameter should be NULL so as to use the default computations.
 * @param bool $mine overridePassword the password check
 * @return bool
 *
 * @return bool
 */
function next_image($all = false, $firstPageCount = NULL, $mine = NULL) {
    global $_zp_images, $_zp_current_image, $_zp_current_album, $_zp_page, $_zp_current_image_restore, $_zp_current_search, $_zp_gallery, $_firstPageImages;
    if (is_null($firstPageCount)) {
        $firstPageCount = $_firstPageImages;
    }
    $imagePageOffset = getTotalPages(2); /* gives us the count of pages for album thumbs */
    if ($all) {
        $imagePage = 1;
        $firstPageCount = 0;
    } else {
        $_firstPageImages = $firstPageCount; /* save this so pagination can see it */
        $imagePage = $_zp_page - $imagePageOffset;
    }
    if ($firstPageCount > 0 && $imagePageOffset > 0) {
        $imagePage = $imagePage + 1; /* can share with last album page */
    }
    if ($imagePage <= 0) {
        return false; /* we are on an album page */
    }
    if (is_null($_zp_images)) {
        if (in_context(ZP_SEARCH)) {
            $_zp_images = $_zp_current_search->getImages($all ? 0 : ($imagePage), $firstPageCount, NULL, NULL, true, $mine);
        } else {
            $_zp_images = $_zp_current_album->getImages($all ? 0 : ($imagePage), $firstPageCount, NULL, NULL, true, $mine);
        }
        if (empty($_zp_images)) {
            return NULL;
        }
        $_zp_current_image_restore = $_zp_current_image;
        $img = array_shift($_zp_images);
        $_zp_current_image = newImage($_zp_current_album, $img, true, true);
        save_context();
        add_context(ZP_IMAGE);
        return true;
    } else if (empty($_zp_images)) {
        $_zp_images = NULL;
        $_zp_current_image = $_zp_current_image_restore;
        restore_context();
        return false;
    } else {
        $img = array_shift($_zp_images);
        $_zp_current_image = newImage($_zp_current_album, $img, true, true);
        return true;
    }
}

//*** Image Context ************************
//******************************************

/**
 * Sets the image passed as the current image
 *
 * @param object $image the image to become current
 */
function makeImageCurrent($image) {
    if (!is_object($image))
        return;
    global $_zp_current_album, $_zp_current_image;
    $_zp_current_image = $image;
    $_zp_current_album = $_zp_current_image->getAlbum();
    set_context(ZP_INDEX | ZP_ALBUM | ZP_IMAGE);
}

/**
 * Returns the raw title of the current image.
 *
 * @return string
 */
function getImageTitle() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return $_zp_current_image->getTitle();
}

/**
 * Returns a text-only title of the current image.
 *
 * @return string
 */
function getBareImageTitle() {
    return getBare(getImageTitle());
}

/**
 * Returns the image title taged with not visible annotation.
 *
 * @return string
 */
function getAnnotatedImageTitle() {
    global $_zp_current_image;
    $title = getBareImageTitle();
    if (!$_zp_current_image->isPublished()) {
        $title .= "\n" . gettext('The image is marked un-published.');
    }
    return $title;
}

function printAnnotatedImageTitle() {
    echo html_encode(getAnnotatedImageTitle());
}

/**
 * Prints title of the current image
 *
 * @author Ozh
 */
function printImageTitle() {
    echo html_encodeTagged(getImageTitle());
}

function printBareImageTitle() {
    echo html_encode(getBareImageTitle());
}

/**
 * Returns the 'n' of n of m images
 *
 * @return int
 */
function imageNumber() {
    global $_zp_current_image, $_zp_current_search, $_zp_current_album;
    $name = $_zp_current_image->getFileName();
    if (in_context(ZP_SEARCH) || (in_context(ZP_SEARCH_LINKED) && !in_context(ZP_ALBUM_LINKED))) {
        $folder = $_zp_current_image->imagefolder;
        $images = $_zp_current_search->getImages();
        $c = 0;
        foreach ($images as $image) {
            $c++;
            if ($name == $image['filename'] && $folder == $image['folder']) {
                return $c;
            }
        }
    } else {
        return $_zp_current_image->getIndex() + 1;
    }
    return false;
}

/**
 * Returns the image date of the current image in yyyy-mm-dd hh:mm:ss format.
 * Pass it a date format string for custom formatting
 *
 * @param string $format formatting string for the data
 * @return string
 */
function getImageDate($format = null) {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    $d = $_zp_current_image->getDateTime();
    if (empty($d) || ($d == '0000-00-00 00:00:00')) {
        return false;
    }
    if (is_null($format)) {
        return $d;
    }
    return zpFormattedDate($format, strtotime($d));
}

/**
 * Prints the date of the current album
 *
 * @param string $before Insert here the text to be printed before the date.
 * @param string $format Format string for the date formatting
 */
function printImageDate($before = '', $format = null) {
    global $_zp_current_image;
    if (is_null($format)) {
        $format = DATE_FORMAT;
    }
    $date = getImageDate($format);
    if ($date) {
        if ($before) {
            $date = '<span class="beforetext">' . $before . '</span>' . $date;
        }
    }
    echo html_encodeTagged($date);
}

// IPTC fields
/**
 * Returns the Location field of the current image
 *
 * @return string
 */
function getImageLocation() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return $_zp_current_image->getLocation();
}

/**
 * Returns the City field of the current image
 *
 * @return string
 */
function getImageCity() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return $_zp_current_image->getcity();
}

/**
 * Returns the State field of the current image
 *
 * @return string
 */
function getImageState() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return $_zp_current_image->getState();
}

/**
 * Returns the Country field of the current image
 *
 * @return string
 */
function getImageCountry() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return $_zp_current_image->getCountry();
}

/**
 * Returns the raw description of the current image.
 * new lines are replaced with <br /> tags
 *
 * @return string
 */
function getImageDesc() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return $_zp_current_image->getDesc();
}

/**
 * Returns a text-only description of the current image.
 *
 * @return string
 */
function getBareImageDesc() {
    return getBare(getImageDesc());
}

/**
 * Prints the description of the current image.
 * Converts and displays line breaks set in the admin field as <br />.
 *
 */
function printImageDesc() {
    echo html_encodeTagged(getImageDesc());
}

function printBareImageDesc() {
    echo html_encode(getBareImageDesc());
}

/**
 * A composit for getting image data
 *
 * @param string $field which field you want
 * @return string
 */
function getImageData($field) {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    return get_language_string($_zp_current_image->get($field));
}

/**
 * Returns the custom_data field of the current image
 *
 * @return string
 */
function getImageCustomData() {
    Global $_zp_current_image;
    return $_zp_current_image->getCustomData();
}

/**
 * Prints the custom_data field of the current image.
 * Converts and displays line breaks set in the admin field as <br />.
 *
 * @return string
 */
function printImageCustomData() {
    $data = getImageCustomData();
    $data = str_replace("\r\n", "\n", $data);
    $data = str_replace("\n", "<br />", $data);
    echo $data;
}

/**
 * Prints arbitrary data from the image object
 *
 * @param string $field the field name of the data desired
 * @param string $label text to label the field.
 * @author Ozh
 */
function printImageData($field, $label = '') {
  global $_zp_current_image;
  $text = getImageData($field);
  if (!empty($text)) {
    echo html_encodeTagged($label . $text);
  }
}

/**
 * Returns the file size of the full original image
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @global obj $_zp_current_image
 * @return int
 */
function getFullImageFilesize() {
    global $_zp_current_image;
    $filesize = $_zp_current_image->getFilesize();
    if($filesize) {
        return byteConvert($filesize);
    }
}

/**
 * True if there is a next image
 *
 * @return bool
 */
function hasNextImage() {
  global $_zp_current_image;
  if (is_null($_zp_current_image))
    return false;
  return $_zp_current_image->getNextImage();
}

/**
 * True if there is a previous image
 *
 * @return bool
 */
function hasPrevImage() {
  global $_zp_current_image;
  if (is_null($_zp_current_image))
    return false;
  return $_zp_current_image->getPrevImage();
}

/**
 * Returns the url of the next image.
 *
 * @return string
 */
function getNextImageURL() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_album, $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    $nextimg = $_zp_current_image->getNextImage();
    return $nextimg->getLink();
}

/**
 * Returns the url of the previous image.
 *
 * @return string
 */
function getPrevImageURL() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_album, $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    $previmg = $_zp_current_image->getPrevImage();
    return $previmg->getLink();
}

/**
 * Returns the thumbnail of the previous image.
 *
 * @return string
 */
function getPrevImageThumb() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    $img = $_zp_current_image->getPrevImage();
    return $img->getThumb();
}

/**
 * Returns the thumbnail of the next image.
 *
 * @return string
 */
function getNextImageThumb() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    $img = $_zp_current_image->getNextImage();
    return $img->getThumb();
}

/**
 * Returns the url of the current image.
 *
 * @return string
 */
function getImageURL() {
    if (!in_context(ZP_IMAGE))
        return false;
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    return $_zp_current_image->getLink();
}

/**
 * Prints the link to the current  image.
 *
 * @param string $text text for the link
 * @param string $title title tag for the link
 * @param string $class optional style class for the link
 * @param string $id optional style id for the link
 */
function printImageURL($text, $title, $class = NULL, $id = NULL) {
    printLinkHTML(getImageURL(), $text, $title, $class, $id);
}

/**
 * Returns the Metadata infromation from the current image
 *
 * @param $image optional image object
 * @param string $displayonly set to true to return only the items selected for display
 * @return array
 */
function getImageMetaData($image = NULL, $displayonly = true) {
    global $_zp_current_image, $_zp_exifvars;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image) || !$image->get('hasMetadata')) {
        return false;
    }
    $data = $image->getMetaData();
    if ($displayonly) {
        foreach ($data as $field => $value) { //    remove the empty or not selected to display
            if (!$value || !$_zp_exifvars[$field][3]) {
                unset($data[$field]);
            }
        }
    }
    if (count($data) > 0) {
        return $data;
    }
    return false;
}

/**
 * Prints the Metadata data of the current image
 *
 * @param string $title title tag for the class
 * @param bool $toggle set to true to get a javascript toggle on the display of the data
 * @param string $id style class id
 * @param string $class style class
 * @author Ozh
 */
function printImageMetadata($title = NULL, $toggle = true, $id = 'imagemetadata', $class = null, $span = NULL) {
    global $_zp_exifvars, $_zp_current_image;
    if (false === ($exif = getImageMetaData($_zp_current_image, true))) {
        return;
    }
    if (is_null($title)) {
        $title = gettext('Image Info');
    }
    if ($class) {
        $class = ' class="' . $class . '"';
    }
    if (!$span) {
        $span = 'exif_link';
    }
    $dataid = $id . '_data';
    if ($id) {
        $id = ' id="' . $id . '"';
    }
    $style = '';
    if ($toggle) {
     if(zp_has_filter('theme_head', 'colorbox::css')) {
        $modal_class = ' colorbox';
        ?>
        <script>
        // <!-- <![CDATA[
        $(document).ready(function () {
            $(".colorbox").colorbox({
                inline: true,
                href: "#imagemetadata",
                close: '<?php echo gettext("close"); ?>'
            });
        });
        // ]]> -->
        </script>
        <?php
     } else {
         $modal_class = '';
            // we only need this eventhanlder if there is no colorbox! 
            ?> 
            <script> 
            // <!-- <![CDATA[
            $(document).ready(function () {
                $(".metadata_toggle").click(function(event) { 
                    event.preventDefault(); $("#<?php echo $dataid; ?>").toggle(); 
                }); 
            });
            // ]]> -->              
            </script> 
            <?php
        }
        $style = ' style="display:none"';
        ?>
        <span id="<?php echo $span; ?>" class="metadata_title">
            <a href="#" class="metadata_toggle<?php echo $modal_class; ?>" title="<?php echo $title; ?>"><?php echo $title; ?></a>
        </span>
        <?php
    } 
    ?>
    <div id="<?php echo $dataid; ?>"<?php echo $style; ?>>
        <div<?php echo $id . $class; ?>>
            <table>
                <?php
                foreach ($exif as $field => $value) {
                    $label = $_zp_exifvars[$field][2];
                    echo "<tr><td class=\"label\">$label:</td><td class=\"value\">";
                    switch ($_zp_exifvars[$field][6]) {
                        case 'time':
                            echo zpFormattedDate(DATE_FORMAT, strtotime($value));
                            break;
                        default:
                            echo html_encode($value);
                            break;
                    }
                    echo "</td></tr>\n";
                }
                ?>
            </table>
        </div>
    </div>
    <?php
}

/**
 * Returns an array with the height & width
 *
 * @param int $size size
 * @param int $width width
 * @param int $height height
 * @param int $cw crop width
 * @param int $ch crop height
 * @param int $cx crop x axis
 * @param int $cy crop y axis
 * @param obj $image The image object for which the size is desired. NULL means the current image
 * @param string $type "image" (sizedimage) (default), "thumb" (thumbnail) required for using option settings for uncropped images
 * @return array
 */
function getSizeCustomImage($size, $width = NULL, $height = NULL, $cw = NULL, $ch = NULL, $cx = NULL, $cy = NULL, $image = NULL, $type = 'image') {
  global $_zp_current_image;
  if (is_null($image))
    $image = $_zp_current_image;
  if (is_null($image))
    return false;
    
  //if we set width/height we are cropping and those are the sizes already
  if (!is_null($width) && !is_null($height)) {
    return array($width, $height);
  }
    switch ($type) {
        case 'thumb':
            $h = $image->getThumbHeight();
            $w = $image->getThumbWidth();
            $thumb = true;
            $side = getOption('thumb_use_side');
            break;
        default:
        case 'image':
            $h = $image->getHeight();
            $w = $image->getWidth();
            $thumb = false;
            if (isImageVideo($image)) { // size is determined by the player
                return array($w, $h);
            }
            $side = getOption('image_use_side');
            break;
    }
    $us = getOption('image_allow_upscale');
  $args = getImageParameters(array($size, $width, $height, $cw, $ch, $cx, $cy, NULL, $thumb, NULL, $thumb, NULL, NULL, NULL), $image->album->name);
  @list($size, $width, $height, $cw, $ch, $cx, $cy, $quality, $thumb, $crop, $thumbstandin, $passedWM, $adminrequest, $effects) = $args;
  if (!empty($size)) {
    $dim = $size;
    $width = $height = false;
  } else if (!empty($width)) {
    $dim = $width;
    $size = $height = false;
  } else if (!empty($height)) {
    $dim = $height;
    $size = $width = false;
  } else {
    $dim = 1;
  }

  if ($w == 0) {
    $hprop = 1;
  } else {
    $hprop = round(($h / $w) * $dim);
  }
  if ($h == 0) {
    $wprop = 1;
  } else {
    $wprop = round(($w / $h) * $dim);
  }
  if (($size && ($side == 'longest' && $h > $w) || ($side == 'height') || ($side == 'shortest' && $h < $w)) || $height) {
// Scale the height
    $newh = $dim;
    $neww = $wprop;
  } else {
// Scale the width
    $neww = $dim;
    $newh = $hprop;
  } 
  if (!$us && $newh >= $h && $neww >= $w) {
    return array($w, $h);
  } else {
    if ($cw && $cw < $neww)
      $neww = $cw;
    if ($ch && $ch < $newh)
      $newh = $ch;
    if ($size && $ch && $cw) {
      $neww = $cw;
      $newh = $ch;
    }
    return array($neww, $newh);
  }
}

/**
 * Returns an array [width, height] of the default-sized image.
 *
 * @param int $size override the 'image_zize' option
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return array
 */
function getSizeDefaultImage($size = NULL, $image = NULL) {
  if (is_null($size))
    $size = getOption('image_size');
  return getSizeCustomImage($size, NULL, NULL, NULL, NULL, NULL, NULL, $image);
}

/**
 * Returns an array [width, height] of the original image.
 *
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return array
 */
function getSizeFullImage($image = NULL) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    return array($image->getWidth(), $image->getHeight());
}

/**
 * The width of the default-sized image (in printDefaultSizedImage)
 *
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return int
 */
function getDefaultWidth($size = NULL, $image = NULL) {
    $size_a = getSizeDefaultImage($size, $image);
    return $size_a[0];
}

/**
 * Returns the height of the default-sized image (in printDefaultSizedImage)
 *
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return int
 */
function getDefaultHeight($size = NULL, $image = NULL) {
    $size_a = getSizeDefaultImage($size, $image);
    return $size_a[1];
}

/**
 * Returns the width of the original image
 *
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return int
 */
function getFullWidth($image = NULL) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    return $image->getWidth();
}

/**
 * Returns the height of the original image
 *
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return int
 */
function getFullHeight($image = NULL) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    return $image->getHeight();
}

/**
 * Returns true if the image is landscape-oriented (width is greater than height) 
 * or - kept here for backwards compatibility - square (equal widht and height)
 * 
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return bool
 */
function isLandscape($image = NULL) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    return ($image->isLandscape() || $image->isSquare());
}

/**
 * Returns the url to the default sized image.
 *
 * @param $image object the image for which the size is desired. NULL means the current image
 *
 * @return string
 */
function getDefaultSizedImage($image = NULL) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    return $image->getSizedImage(getOption('image_size'));
}

/**
 * Show video player with video loaded or display the image.
 *
 * @param string $alt Alt text
 * @param string $class Optional style class
 * @param string $id Optional style id
 * @param string $title Optional title attribute
 * @param obj $image optional image object, null means current image
 */
function printDefaultSizedImage($alt, $class = null, $id = null, $title = null, $image = null) {
    global $_zp_current_image;
    if (is_null($image)) {
        $image = $_zp_current_image;
    }
    if (is_null($image)) {
        return false;
    }
    if (empty($title)) {
        $title = $alt;
    }
    $attr = array(
            'alt' => html_encode($alt),
            'class' => $class,
            'title' => html_encode($title),
            'id' => $id,
            'loading' => 'lazy',
            'width' => getDefaultWidth(),
            'height' => getDefaultHeight()
    );
    if (!$image->isPublished()) {
        $attr['class'] .= " not_visible";
    }
    $album = $image->getAlbum();
    $pwd = $album->getPassword();
    if (!empty($pwd)) {
        $attr['class'] .= " password_protected";
    }
    if (isImagePhoto($image)) { //Print images
        $attr['src'] = html_pathurlencode(getDefaultSizedImage());
        $attr_filtered = zp_apply_filter('standard_image_attr', $attr, $image);
        $attributes = generateAttributesFromArray($attr_filtered);
        $html = '<img' . $attributes . ' />';
        $html = zp_apply_filter('standard_image_html', $html, $image);
        echo $html;
    } else { // better be a plugin class then
        echo $image->getContent();
    }
}

/**
 * Returns the url to the thumbnail of the current image.
 *
 * @return string
 */
function getImageThumb() {
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    return $_zp_current_image->getThumb();
}

/**
 * @param string $alt Alt text
 * @param string $class optional class attribute
 * @param string $id optional id attribute
 * @param string $title optional title attribute
 * @param obj $image optional image object, null means current image
 */
function printImageThumb($alt, $class = null, $id = null, $title = null, $image = null) {
    global $_zp_current_image;
    if (is_null($image)) {
        $image = $_zp_current_image;
    }
    if (is_null($image)) {
        return false;
    }
    if (empty($title)) {
        $title = $alt;
    }
    $attr = array(
            'alt' => html_encode($alt),
            'class' => $class,
            'title' => html_encode($title),
            'id' => $id,
            'loading' => 'lazy'
    );
    if (!$image->isPublished()) {
        $attr['class'] .= " not_visible";
    }
    $album = $image->getAlbum();
    $pwd = $album->getPassword();
    if (!empty($pwd)) {
        $attr['class'] .= " password_protected";
    }
    $attr['src'] = html_pathurlencode($image->getThumb());
    $sizes = getSizeDefaultThumb($image);
    $attr['width'] = $sizes[0];
    $attr['height'] = $sizes[1];
    $attr_filtered = zp_apply_filter('standard_image_thumb_attr', $attr, $image);
    $attributes = generateAttributesFromArray($attr_filtered);
    $html = '<img' . $attributes . ' />';
    $html = zp_apply_filter('standard_image_thumb_html', $html, $image);
    echo $html;
}

/**
 * Gets the width and height of a default thumb for the <img> tag height/width
 * @global type $_zp_current_image
 * @param obj $image Image object, if NULL the current image is used
 * @return aray
 */
function getSizeDefaultThumb($image = NULL) {
    global $_zp_current_image;
    if (is_null($image)) {
        $image = $_zp_current_image;
    }
    $s = getOption('thumb_size');
    if (getOption('thumb_crop')) {
        $w = getOption('thumb_crop_width');
        $h = getOption('thumb_crop_height');
        $sizes = getSizeCustomImage($s, $w, $h, $w, $h, null, null, $image, 'thumb');
    } else {
        $w = $h = $s;
        $sizes = getSizeCustomImage($s, NULL, NULL, NULL, NULL, NULL, NULL, $image, 'thumb');
    }
    return $sizes;
}

/**
 * Returns the url to original image.
 * It will return a protected image is the option "protect_full_image" is set
 *
 * @param $image optional image object
 * @return string
 */
function getFullImageURL($image = NULL) {
    global $_zp_current_image;
    if (is_null($image)) {
        $image = $_zp_current_image;
    }
    if (is_null($image)) {
        return false;
    }
    $outcome = getOption('protect_full_image');
    if ($outcome == 'no-access') {
        return NULL;
    }
    if ($outcome == 'unprotected') {
        return $image->getFullImageURL();
    } else {
        return getProtectedImageURL($image, $outcome);
    }
}

/**
 * Returns the "raw" url to the image in the albums folder
 *
 * @param $image optional image object
 * @return string
 *
 */
function getUnprotectedImageURL($image = NULL) {
    global $_zp_current_image;
    if (is_null($image)) {
        $image = $_zp_current_image;
    }
    if (!is_null($image)) {
        return $image->getFullImageURL();
    }
}

/**
 * Returns an url to the password protected/watermarked current image
 *
 * @param object $image optional image object overrides the current image
 * @param string $disposal set to override the 'protect_full_image' option. 'protected', "download", "unprotected" or "no-access"
 * @return string
 * */
function getProtectedImageURL($image = NULL, $disposal = NULL) {
    global $_zp_current_image;
    if (is_null($disposal)) {
        $disposal = getOption('protect_full_image');
    }
    if ($disposal == 'no-access')
        return NULL;
    if (is_null($image)) {
        if (!in_context(ZP_IMAGE))
            return false;
        if (is_null($_zp_current_image))
            return false;
        $image = $_zp_current_image;
    }
    $album = $image->getAlbum();
    $watermark_use_image = getWatermarkParam($image, WATERMARK_FULL);
    if (!empty($watermark_use_image)) {
        $wmt = $watermark_use_image;
    } else {
        $wmt = false;
    }
    $args = array('FULL', NULL, NULL, NULL, NULL, NULL, NULL, (int) getOption('full_image_quality'), NULL, NULL, NULL, $wmt, false, NULL, NULL);
    $cache_file = getImageCacheFilename($album->name, $image->filename, $args);
    $cache_path = SERVERCACHE . $cache_file;
    if ($disposal != 'download' && OPEN_IMAGE_CACHE && file_exists($cache_path)) {
        return WEBPATH . '/' . CACHEFOLDER . pathurlencode(imgSrcURI($cache_file));
    } else if ($disposal == 'unprotected') {
        return getImageURI($args, $album->name, $image->filename, $image->filemtime);
    } else {
        $params = '&q=' . getOption('full_image_quality');
        if (!empty($watermark_use_image)) {
            $params .= '&wmk=' . $watermark_use_image;
        }
        if ($disposal) {
            $params .= '&dsp=' . $disposal;
        }
        $params .= '&check=' . sha1(HASH_SEED . serialize($args));
        if (is_array($image->filename)) {
            $album = dirname($image->filename['source']);
            $image = basename($image->filename['source']);
        } else {
            $album = $album->name;
            $image = $image->filename;
        }
        return WEBPATH . '/' . ZENFOLDER . '/full-image.php?a=' . $album . '&i=' . $image . $params;
    }
}

/**
 * Returns a link to the current image custom sized to $size
 *
 * @param int $size The size the image is to be
 */
function getSizedImageURL($size) {
    return getCustomImageURL($size);
}

/**
 * Returns the url to the image with the dimensions you define with this function.
 *
 * @param int $size the size of the image to have
 * @param int $width width
 * @param int $height height
 * @param int $cropw crop width
 * @param int $croph crop height
 * @param int $cropx crop part x axis
 * @param int $cropy crop part y axis
 * @param bool $thumbStandin set true to inhibit watermarking
 * @param bool $effects image effects (e.g. set gray to force to grayscale)
 * @return string
 *
 * $size, $width, and $height are used in determining the final image size.
 * At least one of these must be provided. If $size is provided, $width and
 * $height are ignored. If both $width and $height are provided, the image
 * will have those dimensions regardless of the original image height/width
 * ratio. (Yes, this means that the image may be distorted!)
 *
 * The $crop* parameters determine the portion of the original image that
 * will be incorporated into the final image.
 *
 * $cropw and $croph "sizes" are typically proportional. That is you can
 * set them to values that reflect the ratio of width to height that you
 * want for the final image. Typically you would set them to the final
 * height and width. These values will always be adjusted so that they are
 * not larger than the original image dimensions.
 *
 * The $cropx and $cropy values represent the offset of the crop from the
 * top left corner of the image. If these values are provided, the $croph
 * and $cropw parameters are treated as absolute pixels not proportions of
 * the image. If cropx and cropy are not provided, the crop will be
 * "centered" in the image.
 *
 * When $cropx and $cropy are not provided the crop is offset from the top
 * left proportionally to the ratio of the final image size and the crop
 * size.
 *
 * Some typical croppings:
 *
 * $size=200, $width=NULL, $height=NULL, $cropw=200, $croph=100,
 * $cropx=NULL, $cropy=NULL produces an image cropped to a 2x1 ratio which
 * will fit in a 200x200 pixel frame.
 *
 * $size=NULL, $width=200, $height=NULL, $cropw=200, $croph=100, $cropx=100,
 * $cropy=10 will will take a 200x100 pixel slice from (10,100) of the
 * picture and create a 200x100 image
 *
 * $size=NULL, $width=200, $height=100, $cropw=200, $croph=120, $cropx=NULL,
 * $cropy=NULL will produce a (distorted) image 200x100 pixels from a 1x0.6
 * crop of the image.
 *
 * $size=NULL, $width=200, $height=NULL, $cropw=180, $croph=120, $cropx=NULL, $cropy=NULL
 * will produce an image that is 200x133 from a 1.5x1 crop that is 5% from the left
 * and 15% from the top of the image.
 *
 */
function getCustomImageURL($size, $width = NULL, $height = NULL, $cropw = NULL, $croph = NULL, $cropx = NULL, $cropy = NULL, $thumbStandin = false, $effects = NULL) {
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    return $_zp_current_image->getCustomImage($size, $width, $height, $cropw, $croph, $cropx, $cropy, $thumbStandin, $effects);
}

/**
 * Print normal video or custom sized images.
 * Note: a class of 'not_visible' or 'password_protected' will be added as appropriate
 *
 * Notes on cropping:
 *
 * The $crop* parameters determine the portion of the original image that will be incorporated
 * into the final image. The w and h "sizes" are typically proportional. That is you can set them to
 * values that reflect the ratio of width to height that you want for the final image. Typically
 * you would set them to the fincal height and width.
 *
 * @param string $alt Alt text for the url
 * @param int $size size
 * @param int $width width
 * @param int $height height
 * @param int $cropw crop width
 * @param int $croph crop height
 * @param int $cropx crop x axis
 * @param int $cropy crop y axis
 * @param string $class Optional style class
 * @param string $id Optional style id
 * @param bool $thumbStandin set to true to treat as thumbnail
 * @param bool $effects image effects (e.g. set gray to force grayscale)
 * @param string $title Optional title attribute
 * @param string $type "image" (sizedimage) (default), "thumb" (thumbnail) required for using option settings for uncropped images
 * @param obj $image optional image object, null means current image
 * @param bool $maxspace true for maxspace, false default
 */
function printCustomSizedImage($alt, $size, $width = NULL, $height = NULL, $cropw = NULL, $croph = NULL, $cropx = NULL, $cropy = NULL, $class = NULL, $id = NULL, $thumbStandin = false, $effects = NULL, $title = null, $type = 'image', $image = null, $maxspace = false) {
    global $_zp_current_image;
    if (is_null($image)) {
        $image = $_zp_current_image;
    }
    if (is_null($image)) {
        return false;
    }
    if ($maxspace) {
        getMaxSpaceContainer($width, $height, $image);
    }
    if (empty($title)) {
        $title = $alt;
    }
    $attr = array(
            'alt' => html_encode($alt),
            'class' => $class,
            'title' => html_encode($title),
            'id' => $id,
            'loading' => 'lazy'
    );
    if (!$image->isPublished()) {
        $attr['class'] .= " not_visible";
    }
    $album = $image->getAlbum();
    $pwd = $album->getPassword();
    if (!empty($pwd)) {
        $attr['class'] .= " password_protected";
    }
    if ($size && !$maxspace) {
        $type = 'image';
        if ($thumbStandin) {
            $type = 'thumb';
        }
        $dims = getSizeCustomImage($size, null, null, null, null, null, null, $image, $type);
        $attr['width'] = $dims[0];
        $attr['height'] = $dims[1];
    } else {
        $attr['width'] = $width;
        $attr['height'] = $height;
    }
    if (isImagePhoto($image) || $thumbStandin) {
        if ($maxspace) {
            $attr['src'] = html_pathurlencode(getCustomImageURL(null, $width, $height, NULL, NULL, NULL, NULL, $thumbStandin, $effects, null, $image));
        } else {
            $attr['src'] = html_pathurlencode(getCustomImageURL($size, $width, $height, $cropw, $croph, $cropx, $cropy, $thumbStandin, $effects, null, $image));
        }
        $attr_filtered = zp_apply_filter('custom_image_attr', $attr, $image);
        $attributes = generateAttributesFromArray($attr_filtered);
        $html = '<img ' . $attributes . ' />';
        $html = zp_apply_filter('custom_image_html', $html, $thumbStandin, $image);
        echo $html;
    } else { // better be a plugin
        echo $image->getContent($width, $height);
    }
}

/**
 * Returns a link to a un-cropped custom sized version of the current image within the given height and width dimensions.
 * Use for sized images.
 *
 * @param int $width width
 * @param int $height height
 * @return string
 */
function getCustomSizedImageMaxSpace($width, $height) {
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    getMaxSpaceContainer($width, $height, $_zp_current_image);
    return getCustomImageURL(NULL, $width, $height);
}

/**
 * Returns a link to a un-cropped custom sized version of the current image within the given height and width dimensions.
 * Use for sized thumbnails.
 *
 * @param int $width width
 * @param int $height height
 * @return string
 */
function getCustomSizedImageThumbMaxSpace($width, $height) {
    global $_zp_current_image;
    if (is_null($_zp_current_image))
        return false;
    getMaxSpaceContainer($width, $height, $_zp_current_image, true);
    return getCustomImageURL(NULL, $width, $height, NULL, NULL, NULL, NULL, true);
}

/**
 * Creates image thumbnails which will fit un-cropped within the width & height parameters given
 *
 * @param string $alt Alt text for the url
 * @param int $width width
 * @param int $height height
 * @param string $class Optional style class
 * @param string $id Optional style id
 * @param string $title optional title attribute
 * @param obj $image optional image object, null means current image
 */
function printCustomSizedImageThumbMaxSpace($alt, $width, $height, $class = NULL, $id = NULL, $title = null, $image = null) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    printCustomSizedImage($alt, NULL, $width, $height,  NULL,  NULL, NULL,  NULL, $class, $id, true, NULL, $title, 'thumb', $image, true);
}

/**
 * Print normal video or un-cropped within the given height and width dimensions. Use for sized images or thumbnails in an album.
 * Note: a class of 'not_visible' or 'password_protected' will be added as appropriate
 *
 * @param string $alt Alt text for the url
 * @param int $width width
 * @param int $height height
 * @param string $class Optional style class
 * @param string $id Optional style id
 * @param string $title optional title attribute
 * @param obj $image optional image object, null means current image
 */
function printCustomSizedImageMaxSpace($alt, $width, $height, $class = NULL, $id = NULL, $thumb = false, $title = null, $image = null) {
    global $_zp_current_image;
    if (is_null($image))
        $image = $_zp_current_image;
    if (is_null($image))
        return false;
    printCustomSizedImage($alt, NULL, $width, $height,  NULL,  NULL, NULL,  NULL, $class, $id, $thumb, NULL, $title, 'image', $image, true);
}

/**
 * Prints link to an image of specific size
 * @param int $size how big
 * @param string $text URL text
 * @param string $title URL title
 * @param string $class optional URL class
 * @param string $id optional URL id
 */
function printSizedImageURL($size, $text, $title, $class = NULL, $id = NULL) {
    printLinkHTML(getSizedImageURL($size), $text, $title, $class, $id);
}

/**
 * Returns a list of tags for context of the page called where called
 *
 * @return string
 * @since 1.1
 */
function getTags() {
    if (in_context(ZP_IMAGE)) {
        global $_zp_current_image;
        return $_zp_current_image->getTags();
    } else if (in_context(ZP_ALBUM)) {
        global $_zp_current_album;
        return $_zp_current_album->getTags();
    } else if (in_context(ZP_ZENPAGE_PAGE)) {
        global $_zp_current_zenpage_page;
        return $_zp_current_zenpage_page->getTags();
    } else if (in_context(ZP_ZENPAGE_NEWS_ARTICLE)) {
        global $_zp_current_zenpage_news;
        return $_zp_current_zenpage_news->getTags();
    }
    return array();
}

/**
 * Prints a list of tags, editable by admin
 *
 * @param string $option links by default, if anything else the
 *               tags will not link to all other images with the same tag
 * @param string $preText text to go before the printed tags
 * @param string $class css class to apply to the div surrounding the UL list
 * @param string $separator what charactor shall separate the tags
 * @since 1.1
 */
function printTags($option = 'links', $preText = NULL, $class = NULL, $separator = ', ') {
    global $_zp_current_search;
    if (is_null($class)) {
        $class = 'taglist';
    }
    $singletag = getTags();
    $tagstring = implode(', ', $singletag);
    if ($tagstring === '' or $tagstring === NULL) {
        $preText = '';
    }
    if (in_context(ZP_IMAGE)) {
        $object = "image";
    } else if (in_context(ZP_ALBUM)) {
        $object = "album";
    } else if (in_context(ZP_ZENPAGE_PAGE)) {
        $object = "pages";
    } else if (in_context(ZP_ZENPAGE_NEWS_ARTICLE)) {
        $object = "news";
    }
    if (count($singletag) > 0) {
        if (!empty($preText)) {
            echo "<span class=\"tags_title\">" . $preText . "</span>";
        }
        echo "<ul class=\"" . $class . "\">\n";
        if (is_object($_zp_current_search)) {
            $albumlist = $_zp_current_search->getAlbumList();
        } else {
            $albumlist = NULL;
        }
        $ct = count($singletag);
        $x = 0;
        foreach ($singletag as $atag) {
            if (++$x == $ct) {
                $separator = "";
            }
            if ($option === "links") {
                $links1 = "<a href=\"" . html_encode(getSearchURL(search_quote($atag), '', 'tags', 0, array('albums' => $albumlist))) . "\" title=\"" . html_encode($atag) . "\">";
                $links2 = "</a>";
            } else {
                $links1 = $links2 = '';
            }
            echo "\t<li>" . $links1 . $atag . $links2 . $separator . "</li>\n";
        }
        echo "</ul>";
    } else {
        echo "$tagstring";
    }
}

/**
 * Either prints all of the galleries tgs as a UL list or a cloud
 *
 * @param string $option "cloud" for tag cloud, "list" for simple list
 * @param string $class CSS class
 * @param string $sort "results" for relevance list, "random" for random ordering, otherwise the list is alphabetical
 * @param bool $counter TRUE if you want the tag count within brackets behind the tag
 * @param bool $links set to TRUE to have tag search links included with the tag.
 * @param int $maxfontsize largest font size the cloud should display
 * @param int $maxcount the floor count for setting the cloud font size to $maxfontsize
 * @param int $mincount the minimum count for a tag to appear in the output
 * @param int $limit set to limit the number of tags displayed to the top $numtags
 * @param int $minfontsize minimum font size the cloud should display
 * @param bool $exclude_unassigned True or false if you wish to exclude tags that are not assigne to any item (default: true)
 * @param bool $checkaccess True or false (default: false) if you wish to exclude tags that are assigned to items (or are not assigned at all) the visitor is not allowed to see
 * Beware that this may cause overhead on large sites. Usage of the static_html_cache is strongely recommended then.
 * @since 1.1
 */
function printAllTagsAs($option, $class = '', $sort = NULL, $counter = FALSE, $links = TRUE, $maxfontsize = 2, $maxcount = 50, $mincount = 1, $limit = NULL, $minfontsize = 0.8, $exclude_unassigned = true, $checkaccess = false) {
    global $_zp_current_search;
    $option = strtolower($option);
    if ($class != "") {
        $class = ' class="' . $class . '"';
    }
    $tagcount = getAllTagsCount($exclude_unassigned, $checkaccess);
    if (!is_array($tagcount)) {
        return false;
    }
    switch ($sort) {
        case 'results':
            arsort($tagcount);
            if (!is_null($limit)) {
                $tagcount = array_slice($tagcount, 0, $limit);
            }
            break;
        case 'random':
            if (!is_null($limit)) {
                $tagcount = array_slice($tagcount, 0, $limit);
            }
            shuffle_assoc($tagcount);
            break;
        default:
            break;
    }
    ?>
    <ul<?php echo $class; ?>>
        <?php
        if (count($tagcount) > 0) {
            foreach ($tagcount as $key => $val) {
                if (!$counter) {
                    $counter = "";
                } else {
                    $counter = " (" . $val . ") ";
                }
                if ($option == "cloud") { // calculate font sizes, formula from wikipedia
                    if ($val <= $mincount) {
                        $size = $minfontsize;
                    } else {
                        $size = min(max(round(($maxfontsize * ($val - $mincount)) / ($maxcount - $mincount), 2), $minfontsize), $maxfontsize);
                    }
                    $size = str_replace(',', '.', $size);
                    $size = ' style="font-size:' . $size . 'em;"';
                } else {
                    $size = '';
                }
                if ($val >= $mincount) {
                    if ($links) {
                        if (is_object($_zp_current_search)) {
                            $albumlist = $_zp_current_search->getAlbumList();
                        } else {
                            $albumlist = NULL;
                        }
                        $link = getSearchURL(search_quote($key), '', 'tags', 0, array('albums' => $albumlist));
                        ?>
                        <li>
                            <a href="<?php echo html_encode($link); ?>"<?php echo $size; ?>><?php echo $key . $counter; ?></a>
                        </li>
                        <?php
                    } else {
                        ?>
                        <li<?php echo $size; ?>><?php echo $key . $counter; ?></li>
                        <?php
                    }
                }
            } // while end
        } else {
            ?>
            <li><?php echo gettext('No popular tags'); ?></li>
            <?php
        }
        ?>
    </ul>
    <?php
}

/**
 * Retrieves a list of all unique years & months from the images in the gallery
 *
 * @param string $order set to 'desc' for the list to be in descending order
 * @return array
 */
function getAllDates($order = 'asc') {
    $alldates = array();
    $cleandates = array();
    $sql = "SELECT `date` FROM " . prefix('images');
    if (!zp_loggedin()) {
        $sql .= " WHERE `show` = 1";
    }
    $hidealbums = getNotViewableAlbums();
    if (!is_null($hidealbums)) {
        if (zp_loggedin()) {
            $sql .= ' WHERE ';
        } else {
            $sql .= ' AND ';
        }
        foreach ($hidealbums as $id) {
            $sql .= '`albumid`!=' . $id . ' AND ';
        }
        $sql = substr($sql, 0, -5);
    }
    $result = query($sql);
    if ($result) {
        while ($row = db_fetch_assoc($result)) {
            $alldates[] = $row['date'];
        }
        db_free_result($result);
    }
    foreach ($alldates as $adate) {
        if (!empty($adate)) {
            $cleandates[] = substr($adate, 0, 7) . "-01";
        }
    }
    $datecount = array_count_values($cleandates);
    if ($order == 'desc') {
        krsort($datecount);
    } else {
        ksort($datecount);
    }
    return $datecount;
}

/**
 * Prints a compendum of dates and links to a search page that will show results of the date
 *
 * @param string $class optional class
 * @param string $yearid optional class for "year"
 * @param string $monthid optional class for "month"
 * @param string $order set to 'desc' for the list to be in descending order
 */
function printAllDates($class = 'archive', $yearid = 'year', $monthid = 'month', $order = 'asc') {
    global $_zp_current_search, $_zp_gallery_page;
    if (empty($class)) {
        $classactive = 'archive_active';
    } else {
        $classactive = $class . '_active';
        $class = 'class="' . $class . '"';
    }
    if ($_zp_gallery_page == 'search.php') {
        $activedate = getSearchDate('%Y-%m');
    } else {
        $activedate = '';
    }
    if (!empty($yearid)) {
        $yearid = 'class="' . $yearid . '"';
    }
    if (!empty($monthid)) {
        $monthid = 'class="' . $monthid . '"';
    }
    $datecount = getAllDates($order);
    $lastyear = "";
    echo "\n<ul $class>\n";
    $nr = 0;
    foreach($datecount as $key => $val) {
        $nr++;
        if ($key == '0000-00-01') {
            $year = "no date";
            $month = "";
        } else {
            $dt = strftime('%Y-%B', strtotime($key));
            $year = substr($dt, 0, 4);
            $month = substr($dt, 5);
        }

        if ($lastyear != $year) {
            $lastyear = $year;
            if ($nr != 1) {
                echo "</ul>\n</li>\n";
            }
            echo "<li $yearid>$year\n<ul $monthid>\n";
        }
        if (is_object($_zp_current_search)) {
            $albumlist = $_zp_current_search->getAlbumList();
        } else {
            $albumlist = NULL;
        }
        $datekey = substr($key, 0, 7);
        if ($activedate = $datekey) {
            $cl = ' class="' . $classactive . '"';
        } else {
            $cl = '';
        }
        echo '<li' . $cl . '><a href="' . html_encode(getSearchURL('', $datekey, '', 0, array('albums' => $albumlist))) . '">' . $month . ' (' . $val . ')</a></li>' . "\n";
    }
    echo "</ul>\n</li>\n</ul>\n";
}

/**
 * Produces the url to a custom page (e.g. one that is not album.php, image.php, or index.php)
 *
 * @param string $page page name to include in URL
 * @param string $q query string to add to url
 * @param bool $webpath host path to be prefixed. If "false" is passed you will get a localized "WEBPATH"
 * @return string
 */
function getCustomPageURL($page, $q = '', $webpath = null) {
    global $_zp_conf_vars;
    if (array_key_exists($page, $_zp_conf_vars['special_pages'])) {
        $rewrite = preg_replace('~^_PAGE_/~', _PAGE_ . '/', $_zp_conf_vars['special_pages'][$page]['rewrite']) . '/';
    } else {
        $rewrite = '/' . _PAGE_ . '/' . $page . '/';
    }
    $plain = "index.php?p=$page";
    if (!empty($q)) {
        $rewrite .= "?$q";
        $plain .= "&$q";
    }
    return zp_apply_filter('getLink', rewrite_path($rewrite, $plain, $webpath), $page . '.php', null);
}

/**
 * Prints the url to a custom page (e.g. one that is not album.php, image.php, or index.php)
 *
 * @param string $linktext Text for the URL
 * @param string $page page name to include in URL
 * @param string $q query string to add to url
 * @param string $prev text to insert before the URL
 * @param string $next text to follow the URL
 * @param string $class optional class
 */
function printCustomPageURL($linktext, $page, $q = '', $prev = '', $next = '', $class = NULL) {
    if (!is_null($class)) {
        $class = 'class="' . $class . '"';
    }
    echo $prev . "<a href=\"" . html_encode(getCustomPageURL($page, $q)) . "\" $class title=\"" . html_encode($linktext) . "\">" . html_encode($linktext) . "</a>" . $next;
}

//*** Search functions *******************************************************
//****************************************************************************

/**
 * tests if a search page is an "archive" page
 *
 * @return bool
 */
function isArchive() {
    return isset($_REQUEST['date']);
}

/**
 * Returns a search URL
 *
 * @param mixed $words the search words target
 * @param mixed $dates the dates that limit the search
 * @param mixed $fields the fields on which to search
 * @param int $page the page number for the URL
 * @param array $object_list the list of objects to search
 * @return string
 * @since 1.1.3
 */
function getSearchURL($words, $dates, $fields, $page, $object_list = NULL) {
    $urls = '';
    $rewrite = false;
    if (MOD_REWRITE) {
        $rewrite = true;
        if (is_array($object_list)) {
            foreach ($object_list as $obj) {
                if ($obj) {
                    $rewrite = false;
                    break;
                }
            }
        }
    }

    if ($rewrite) {
        if (empty($dates)) {
            $url = SEO_WEBPATH . '/' . _SEARCH_ . '/';
        } else {
            $url = SEO_WEBPATH . '/' . _ARCHIVE_ . '/';
        }
    } else {
        $url = SEO_WEBPATH . "/index.php?p=search";
    }
    if (!empty($fields) && empty($dates)) {
        if (!is_array($fields)) {
            $fields = explode(',', $fields);
        }
        $temp = $fields;
        if ($rewrite && count($fields) == 1 && array_shift($temp) == 'tags') {
            $url = SEO_WEBPATH . '/' . _TAGS_ . '/';
        } else {
            $search = new SearchEngine();
            $urls = $search->getSearchFieldsText($fields, 'searchfields=');
        }
    }

    if (!empty($words)) {
        if (is_array($words)) {
            foreach ($words as $key => $word) {
                $words[$key] = search_quote($word);
            }
            $words = implode(',', $words);
        }
        $words = strtr($words, array('%' => '__25__', '&' => '__26__', '#' => '__23__', '/' => '__2F__'));
        if ($rewrite) {
            $url .= urlencode($words) . '/';
        } else {
            $url .= "&words=" . urlencode($words);
        }
    }
    if (!empty($dates)) {
        if (is_array($dates)) {
            $dates = implode(',', $dates);
        }
        if ($rewrite) {
            $url .= $dates . '/';
        } else {
            $url .= "&date=$dates";
        }
    }
    if ($page > 1) {
        if ($rewrite) {
            $url .= "$page/";
        } else {
            if ($urls) {
                $urls .= '&';
            }
            $urls .= "page=$page";
        }
    }
    if (!empty($urls)) {
        if ($rewrite) {
            $url .= '?' . $urls;
        } else {
            $url .= '&' . $urls;
        }
    }
    if (is_array($object_list)) {
        foreach ($object_list as $key => $list) {
            if (!empty($list)) {
                $url .= '&in' . $key . '=' . html_encode(implode(',', $list));
            }
        }
    }
    return $url;
}

/**
 * Prints the search form
 *
 * Search works on a list of tokens entered into the search form.
 *
 * Tokens may be part of boolean expressions using &, |, !, and parens. (Comma is retained as a synonom of | for
 * backwords compatibility.)
 *
 * Tokens may be enclosed in quotation marks to create exact pattern matches or to include the boolean operators and
 * parens as part of the tag..
 *
 * @param string $prevtext text to go before the search form
 * @param string $id css id for the search form, default is 'search'
 * @param string $buttonSource optional path to the image for the button or if not a path to an image,
 *                                          this will be the button hint
 * @param string $buttontext optional text for the button ("Search" will be the default text)
 * @param string $iconsource optional theme based icon for the search fields toggle
 * @param array $query_fields override selection for enabled fields with this list
 * @param array $objects_list optional array of things to search eg. [albums]=>[list], etc.
 *                                                      if the list is simply 0, the objects will be omitted from the search
 * @param string $within set to true to search within current results, false to search fresh
 * @since 1.1.3
 */
function printSearchForm($prevtext = NULL, $id = 'search', $buttonSource = NULL, $buttontext = '', $iconsource = NULL, $query_fields = NULL, $object_list = NULL, $within = NULL) {
    global $_zp_adminJS_loaded, $_zp_current_search;
    $engine = new SearchEngine();
    if (!is_null($_zp_current_search) && !$_zp_current_search->getSearchWords()) {
        $engine->clearSearchWords();
    }
    if (!is_null($object_list)) {
        if (array_key_exists(0, $object_list)) { // handle old form albums list
            trigger_error(gettext('printSearchForm $album_list parameter is deprecated. Pass array("albums"=>array(album, album, ...)) instead.'), E_USER_NOTICE);
            $object_list = array('albums' => $object_list);
        }
    }
    if (empty($buttontext)) {
        $buttontext = gettext("Search");
    }
    $zf = WEBPATH . "/" . ZENFOLDER;
    $searchwords = $engine->codifySearchString();
    if (substr($searchwords, -1, 1) == ',') {
        $searchwords = substr($searchwords, 0, -1);
    }
    $hint = $hintJS = '%s';
    if (empty($searchwords)) {
        $within = false;
    } else {
        $hintJS = gettext('%s within previous results');
    }
    if (is_null($within)) {
        $within = getOption('search_within');
    }
    if ($within) {
        $hint = gettext('%s within previous results');
    }
    if (preg_match('!\/(.*)[\.png|\.jpg|\.jpeg|\.gif]$!', $buttonSource)) {
        $buttonSource = 'src="' . $buttonSource . '" alt="' . $buttontext . '"';
        $button = 'title="' . sprintf($hint, $buttontext) . '"';
        $type = 'image';
    } else {
        $type = 'submit';
        if ($buttonSource) {
            $button = 'value="' . $buttontext . '" title="' . sprintf($hint, $buttonSource) . '"';
            $buttonSource = '';
        } else {
            $button = 'value="' . $buttontext . '" title="' . sprintf($hint, $buttontext) . '"';
        }
    }
    if (empty($iconsource)) {
        $iconsource = WEBPATH . '/' . ZENFOLDER . '/images/searchfields_icon.png';
    }
    if (MOD_REWRITE) {
        $searchurl = SEO_WEBPATH . '/' . _SEARCH_ . '/';
    } else {
        $searchurl = WEBPATH . "/index.php?p=search";
    }
    if (!$within) {
        $engine->clearSearchWords();
    }

    $fields = $engine->allowedSearchFields();
    if (!$_zp_adminJS_loaded) {
        $_zp_adminJS_loaded = true;
        ?>
        <script type="text/javascript" src="<?php echo WEBPATH . '/' . ZENFOLDER; ?>/js/admin.js"></script>
        <?php
    }
    ?>
    <div id="<?php echo $id; ?>">
        <!-- search form -->
        <form method="post" action="<?php echo $searchurl; ?>" id="search_form">
            <script type="text/javascript">
            // <!-- <![CDATA[
            var within = <?php echo (int) $within; ?>;
            function search_(way) {
                within = way;
                if (way) {
                    $('#search_submit').attr('title', '<?php echo sprintf($hintJS, $buttontext); ?>');
                } else {
                    lastsearch = '';
                    $('#search_submit').attr('title', '<?php echo $buttontext; ?>');
                }
                $('#search_input').val('');
            }
            $('#search_form').submit(function() {
                if (within) {
                    var newsearch = $.trim($('#search_input').val());
                    if (newsearch.substring(newsearch.length - 1) == ',') {
                        newsearch = newsearch.substr(0, newsearch.length - 1);
                    }
                    if (newsearch.length > 0) {
                        $('#search_input').val('(<?php echo $searchwords; ?>) AND (' + newsearch + ')');
                    } else {
                        $('#search_input').val('<?php echo $searchwords; ?>');
                    }
                }
                return true;
            });
    $(document).ready(function() {
      $( $("#checkall_searchfields") ).on( "click", function() {
        $("#searchextrashow :checkbox").prop("checked", $("#checkall_searchfields").prop("checked") );
      });
    });
            // ]]> -->
            </script>
            <?php echo $prevtext; ?>
            <div>
                <span class="tagSuggestContainer">
                    <input type="text" name="words" value="" id="search_input" size="10" />
                </span>
                <?php if (count($fields) > 1 || $searchwords) { ?>
                    <a class="toggle_searchextrashow" href="#"><img src="<?php echo $iconsource; ?>" title="<?php echo gettext('search options'); ?>" alt="<?php echo gettext('fields'); ?>" id="searchfields_icon" /></a>
                    <script>
                        $(".toggle_searchextrashow").click(function(event) {
                            event.preventDefault();
                            $("#searchextrashow").toggle();
                        });
                    </script>
                <?php } ?>
                <input type="<?php echo $type; ?>" <?php echo $button; ?> class="button buttons" id="search_submit" <?php echo $buttonSource; ?> data-role="none" />
                <?php
                if (is_array($object_list)) {
                    foreach ($object_list as $key => $list) {
                        ?>
                        <input type="hidden" name="in<?php echo $key ?>" value="<?php
                        if (is_array($list))
                            echo html_encode(implode(',', $list));
                        else
                            echo html_encode($list);
                        ?>" />
                                     <?php
                                 }
                             }
                             ?>
                <br />
                <?php
                if (count($fields) > 1 || $searchwords) {
                    $fields = array_flip($fields);
                    sortArray($fields);
                    $fields = array_flip($fields);
                    if (is_null($query_fields)) {
                        $query_fields = $engine->parseQueryFields();
                    } else {
                        if (!is_array($query_fields)) {
                            $query_fields = $engine->numericFields($query_fields);
                        }
                    }
                    if (count($query_fields) == 0) {
                        $query_fields = $engine->allowedSearchFields();
                    }
                    ?>
                    <div style="display:none;" id="searchextrashow">
                        <?php
                        if ($searchwords) {
                            ?>
                            <label>
                                <input type="radio" name="search_within" id="search_within-1" value="1"<?php if ($within) echo ' checked="checked"'; ?> onclick="search_(1);" />
                                <?php echo gettext('Within'); ?>
                            </label>
                            <label>
                                <input type="radio" name="search_within" id="search_within-0" value="1"<?php if (!$within) echo ' checked="checked"'; ?> onclick="search_(0);" />
                                <?php echo gettext('New'); ?>
                            </label>
                            <?php
                        }
                        if (count($fields) > 1) {
                            ?>
                            <ul>
        <li><label><input type="checkbox" name="checkall_searchfields" id="checkall_searchfields" checked="checked">* <?php echo gettext('Check/uncheck all'); ?> *</label></li>
                                <?php
                                foreach ($fields as $display => $key) {
                                    echo '<li><label><input id="SEARCH_' . html_encode($key) . '" name="SEARCH_' . html_encode($key) . '" type="checkbox"';
                                    if (in_array($key, $query_fields)) {
                                        echo ' checked="checked" ';
                                    }
                                    echo ' value="' . html_encode($key) . '"  /> ' . html_encode($display) . "</label></li>" . "\n";
                                }
                                ?>
                            </ul>
                            <?php
                        }
                        ?>
                    </div>
                    <?php
                }
                ?>
            </div>
        </form>
    </div><!-- end of search form -->
    <?php
}

/**
 * Returns the a sanitized version of the search string
 *
 * @return string
 * @since 1.1
 */
function getSearchWords() {
    global $_zp_current_search;
    if (!in_context(ZP_SEARCH))
        return '';
    return stripcslashes($_zp_current_search->codifySearchString());
}

/**
 * Returns the date of the search
 *
 * @param string $format formatting of the date, default 'F Y'
 * @return string
 * @since 1.1
 */
function getSearchDate($format = '%B %Y') {
    if (in_context(ZP_SEARCH)) {
        global $_zp_current_search;
        $date = $_zp_current_search->getSearchDate();
        if (empty($date)) {
            return "";
        }
        if ($date == '0000-00') {
            return gettext("no date");
        };
        $dt = strtotime($date . "-01");
        return zpFormattedDate($format, $dt);
    }
    return false;
}

/**
 * controls the thumbnail layout of themes.
 *
 * Uses the theme options:
 *  albums_per_row
 *  albums_per_page
 *  images_per_row
 *  images_per_page
 *
 * Computes a normalized images/albums per page and computes the number of
 * images that will fit on the "transitional" page between album thumbs and
 * image thumbs. This function is "internal" and is called from the root
 * index.php script before the theme script is loaded.
 */
function setThemeColumns() {
    global $_zp_current_album, $_firstPageImages, $_oneImagePage;
    $_firstPageImages = false;
    if (($albumColumns = getOption('albums_per_row')) <= 1)
        $albumColumns = false;
    if (($imageColumns = getOption('images_per_row')) <= 1)
        $imageColumns = false;
    $albcount = max(1, getOption('albums_per_page'));
    if (($albumColumns) && (($albcount % $albumColumns) != 0)) {
        setOption('albums_per_page', $albcount = ((floor($albcount / $albumColumns) + 1) * $albumColumns), false);
    }
    $imgcount = max(1, getOption('images_per_page'));
    if (($imageColumns) && (($imgcount % $imageColumns) != 0)) {
        setOption('images_per_page', $imgcount = ((floor($imgcount / $imageColumns) + 1) * $imageColumns), false);
    }
    if ((getOption('thumb_transition') && !$_oneImagePage) && in_context(ZP_ALBUM | ZP_SEARCH) && $albumColumns && $imageColumns) {
        $count = getNumAlbums();
        if ($count == 0) {
            $_firstPageImages = 0;
        }
        $rowssused = ceil(($count % $albcount) / $albumColumns); /* number of album rows unused */
        $leftover = floor(max(1, getOption('images_per_page')) / $imageColumns) - $rowssused;
        $_firstPageImages = max(0, $leftover * $imageColumns); /* number of images that fill the leftover rows */
        if ($_firstPageImages == $imgcount) {
            $_firstPageImages = 0;
        }
    }
}

//************************************************************************************************
// album password handling
//************************************************************************************************

/**
 * returns the auth type of a guest login
 *
 * @param string $hint
 * @param string $show
 * @return string
 */
function checkForGuest(&$hint = NULL, &$show = NULL) {
    global $_zp_gallery, $_zp_gallery_page, $_zp_current_zenpage_page, $_zp_current_category, $_zp_current_zenpage_news;
    $authType = zp_apply_filter('checkForGuest', NULL);
    if (!is_null($authType))
        return $authType;
    if (in_context(ZP_SEARCH)) { // search page
        $hash = getOption('search_password');
        if (getOption('search_user') != '')
            $show = true;
        $hint = get_language_string(getOption('search_hint'));
        $authType = 'zpcms_auth_search';
        if (empty($hash)) {
            $hash = $_zp_gallery->getPassword();
            if ($_zp_gallery->getUser() != '')
                $show = true;
            $hint = $_zp_gallery->getPasswordHint();
            $authType = 'zpcms_auth_gallery';
        }
        if (!empty($hash) && zp_getCookie($authType) == $hash) {
            return $authType;
        }
    } else if (!is_null($_zp_current_zenpage_news)) {
        $authType = $_zp_current_zenpage_news->checkAccess($hint, $show);
        return $authType;
    } else if (isset($_GET['album'])) { // album page
        list($album, $image) = rewrite_get_album_image('album', 'image');
        if ($authType = checkAlbumPassword($album, $hint)) {
            return $authType;
        } else {
            $alb = newAlbum($album);
            if ($alb->getUser() != '')
                $show = true;
            return false;
        }
    } else { // other page
        $hash = $_zp_gallery->getPassword();
        if ($_zp_gallery->getUser() != '')
            $show = true;
        $hint = $_zp_gallery->getPasswordHint();
        if (!empty($hash) && zp_getCookie('zpcms_auth_gallery') == $hash) {
            return 'zpcms_auth_gallery';
        }
    }
    if (empty($hash))
        return 'zp_public_access';
    return false;
}

/**
 * Checks to see if a password is needed
 *
 * Returns true if access is allowed
 *
 * The password protection is hereditary. This normally only impacts direct url access to an object since if
 * you are going down the tree you will be stopped at the first place a password is required.
 *
 *
 * @param string $hint the password hint
 * @param bool $show whether there is a user associated with the password.
 * @return bool
 * @since 1.1.3
 */
function checkAccess(&$hint = NULL, &$show = NULL) {
    global $_zp_current_album, $_zp_current_search, $_zp_gallery, $_zp_gallery_page,
    $_zp_current_zenpage_page, $_zp_current_zenpage_news;
    if (GALLERY_SECURITY != 'public') // only registered users allowed
        $show = true; //    therefore they will need to supply their user id is something fails below

    if ($_zp_gallery->isUnprotectedPage(stripSuffix($_zp_gallery_page)))
        return true;
    if (zp_loggedin()) {
        $fail = zp_apply_filter('isMyItemToView', NULL);
        if (!is_null($fail)) { //   filter had something to say about access, honor it
            return $fail;
        }
        switch ($_zp_gallery_page) {
            case 'album.php':
            case 'image.php':
                if ($_zp_current_album->isMyItem(LIST_RIGHTS)) {
                    return true;
                }
                break;
            case 'search.php':
                if (zp_loggedin(VIEW_SEARCH_RIGHTS)) {
                    return true;
                }
                break;
            default:
                if (zp_loggedin(VIEW_GALLERY_RIGHTS)) {
                    return true;
                }
                break;
        }
    }
    if (GALLERY_SECURITY == 'public' && ($access = checkForGuest($hint, $show))) {
        return $access; // public page or a guest is logged in
    }
    return false;
}

/**
 * Returns a redirection link for the password form
 *
 * @return string
 */
function getPageRedirect() {
  global $_zp_login_error, $_zp_password_form_printed, $_zp_current_search, $_zp_gallery_page,
  $_zp_current_album, $_zp_current_image, $_zp_current_zenpage_news;
    if($_zp_login_error !== 2) {
        return false;
    }
  switch ($_zp_gallery_page) {
    case 'index.php':
      $action = '/index.php';
      break;
    case 'album.php':
      $action = '/index.php?userlog=1&album=' . pathurlencode($_zp_current_album->name);
      break;
    case 'image.php':
      $action = '/index.php?userlog=1&album=' . pathurlencode($_zp_current_album->name) . '&image=' . urlencode($_zp_current_image->filename);
      break;
    case 'pages.php':
      $action = '/index.php?userlog=1&p=pages&title=' . urlencode(getPageTitlelink());
      break;
    case 'news.php':
      $action = '/index.php?userlog=1&p=news';
      if (!is_null($_zp_current_zenpage_news)) {
        $action .= '&title=' . urlencode($_zp_current_zenpage_news->getTitlelink());
      }
      break;
    case 'password.php':
      $action = str_replace(SEO_WEBPATH, '', getRequestURI());
      if ($action == '/' . _PAGE_ . '/password' || $action == '/index.php?p=password') {
        $action = '/index.php';
      }
      break;
    default:
      if (in_context(ZP_SEARCH)) {
        $action = '/index.php?userlog=1&p=search' . $_zp_current_search->getSearchParams();
      } else {
        $action = '/index.php?userlog=1&p=' . substr($_zp_gallery_page, 0, -4);
      }
  }
  return SEO_WEBPATH . $action;
}

/**
 * Prints the album password form
 *
 * @param string $hint hint to the password
 * @param bool $showProtected set false to supress the password protected message
 * @param bool $showuser set true to force the user name filed to be present
 * @param string $redirect optional URL to send the user to after successful login
 *
 * @since 1.1.3
 */
function printPasswordForm($_password_hint, $_password_showuser = NULL, $_password_showProtected = true, $_password_redirect = NULL) {
    global $_zp_login_error, $_zp_password_form_printed, $_zp_current_search, $_zp_gallery, $_zp_gallery_page,
    $_zp_current_album, $_zp_current_image, $theme, $_zp_current_zenpage_page, $_zp_authority;
    if ($_zp_password_form_printed)
        return;
    $_zp_password_form_printed = true;

    if (is_null($_password_redirect))
        $_password_redirect = getPageRedirect();

    if (is_null($_password_showuser))
        $_password_showuser = $_zp_gallery->getUserLogonField();
    ?>
    <div id="passwordform">
        <?php
            if(zp_loggedin()) {
                echo '<p><strong>' . gettext('You are successfully logged in.') . '</strong></p>';
            } else {
                if ($_password_showProtected && !$_zp_login_error) {
                    ?>
                    <p>
                        <?php echo gettext("The page you are trying to view is password protected."); ?>
                    </p>
                    <?php
                }
                if ($loginlink = zp_apply_filter('login_link', NULL)) {
                    $logintext = gettext('login');
                    ?>
                    <a href="<?php echo $loginlink; ?>" title="<?php echo $logintext; ?>"><?php echo $logintext; ?></a>
                    <?php
                } else {
                    $_zp_authority->printLoginForm($_password_redirect, false, $_password_showuser, false, $_password_hint);
                }
            }
        ?>
    </div>
    <?php
}

/**
 * prints the zenphoto logo and link
 *
 */
function printZenphotoLink() {
    echo gettext("Powered by <a href=\"http://www.zenphoto.org\" title=\"The simpler media website CMS\">Zenphoto</a>");
}

/**
 * Expose some informations in a HTML comment
 *
 * @param string $obj the path to the page being loaded
 * @param array $plugins list of activated plugins
 * @param string $theme The theme being used
 */
function exposeZenPhotoInformations($obj = '', $plugins = '', $theme = '') {
    global $_zp_filters;
    $a = basename($obj);
    if ($a != 'full-image.php') {
        echo "\n<!-- zenphoto version " . ZENPHOTO_VERSION;
        if (TEST_RELEASE) {
            echo " THEME: " . $theme . " (" . $a . ")";
            $graphics = zp_graphicsLibInfo();
            $graphics = sanitize(str_replace('<br />', ', ', $graphics['Library_desc']), 3);
            echo " GRAPHICS LIB: " . $graphics . " { memory: " . INI_GET('memory_limit') . " }";
            echo ' PLUGINS: ';
            if (count($plugins) > 0) {
                sort($plugins);
                foreach ($plugins as $plugin) {
                    echo $plugin . ' ';
                }
            } else {
                echo 'none ';
            }
        }
        echo " -->";
    }
}

/**
 * Gets the content of a codeblock for an image, album or Zenpage newsarticle or page.
 *
 * The priority for codeblocks will be (based on context)
 *  1: articles
 *  2: pages
 *  3: images
 *  4: albums
 *  5: gallery.
 *
 * This means, for instance, if we are in ZP_ZENPAGE_NEWS_ARTICLE context we will use the news article
 * codeblock even if others are available.
 *
 * Note: Echoing this array's content does not execute it. Also no special chars will be escaped.
 * Use printCodeblock() if you need to execute script code.
 *
 * @param int $number The codeblock you want to get
 * @param mixed $what optonal object for which you want the codeblock
 *
 * @return string
 */
function getCodeblock($number = 1, $object = NULL) {
    global $_zp_current_album, $_zp_current_image, $_zp_current_zenpage_news, $_zp_current_zenpage_page, $_zp_gallery, $_zp_gallery_page;
    if (!$number) {
        setOptionDefault('codeblock_first_tab', 0);
    }
    if (!is_object($object)) {
        if ($_zp_gallery_page == 'index.php') {
            $object = $_zp_gallery;
        }
        if (in_context(ZP_ALBUM)) {
            $object = $_zp_current_album;
        }
        if (in_context(ZP_IMAGE)) {
            $object = $_zp_current_image;
        }
        if (in_context(ZP_ZENPAGE_PAGE)) {
            if ($_zp_current_zenpage_page->checkAccess()) {
                $object = $_zp_current_zenpage_page;
            }
        }
        if (in_context(ZP_ZENPAGE_NEWS_ARTICLE)) {
            if ($_zp_current_zenpage_news->checkAccess()) {
                $object = $_zp_current_zenpage_news;
            }
        }
    }
    if (!is_object($object)) {
        return NULL;
    }
    $codeblock = getSerializedArray($object->getcodeblock());
    $codeblock = zp_apply_filter('codeblock', @$codeblock[$number], $object, $number);
    if ($codeblock) {
        $codeblock = applyMacros($codeblock);
    }
    return $codeblock;
}

/**
 * Prints the content of a codeblock for an image, album or Zenpage newsarticle or page.
 *
 * @param int $number The codeblock you want to get
 * @param mixed $what optonal object for which you want the codeblock
 *
 * @return string
 */
function printCodeblock($number = 1, $what = NULL) {
    $codeblock = getCodeblock($number, $what);
    if ($codeblock) {
        $context = get_context();
        eval('?>' . $codeblock);
        set_context($context);
    }
}

/**
 * Checks for URL page out-of-bounds for "standard" themes
 * Note: This function assumes that an "index" page will display albums
 * and the pagination be determined by them. Any other "index" page strategy needs to be
 * handled by the theme itself.
 *
 * @param boolean $request
 * @param string $gallery_page
 * @param int $page
 * @return boolean will be true if all is well, false if a 404 error should occur
 */
function checkPageValidity($request, $gallery_page, $page) {
    global $_zp_gallery, $_firstPageImages, $_oneImagePage, $_zp_zenpage, $_zp_current_category;
    $count = NULL;
    switch ($gallery_page) {
        case 'album.php':
        case 'search.php':
            $albums_per_page = max(1, getOption('albums_per_page'));
            $pageCount = (int) ceil(getNumAlbums() / $albums_per_page);
            $imageCount = getNumImages();
            if ($_oneImagePage) {
                if ($_oneImagePage === true) {
                    $imageCount = min(1, $imageCount);
                } else {
                    $imageCount = 0;
                }
            }
            $images_per_page = max(1, getOption('images_per_page'));
            $count = ($pageCount + (int) ceil(($imageCount - $_firstPageImages) / $images_per_page));
            break;
        case 'index.php':
            if (galleryAlbumsPerPage() != 0) {
                $count = (int) ceil($_zp_gallery->getNumAlbums() / galleryAlbumsPerPage());
            }
            break;
        case 'news.php':
            if (in_context(ZP_ZENPAGE_NEWS_CATEGORY)) {
                $count = count($_zp_current_category->getArticles());
            } else {
                $count = count($_zp_zenpage->getArticles());
            }
            $count = (int) ceil($count / ZP_ARTICLES_PER_PAGE);
            break;
        default:
            $count = zp_apply_filter('checkPageValidity', NULL, $gallery_page, $page);
            break;
    }
    if ($page > $count) {
        $request = false; //    page is out of range
    }

    return $request;
}

function print404status($album, $image, $obj) {
    global $_zp_page;
    echo "\n<strong>" . gettext("Zenphoto Error:</strong> the requested object was not found.");
    if (isset($album)) {
        echo '<br />' . sprintf(gettext('Album: %s'), html_encode($album));

        if (isset($image)) {
            echo '<br />' . sprintf(gettext('Image: %s'), html_encode($image));
        }
    } else {
        echo '<br />' . sprintf(gettext('Page: %s'), html_encode(substr(basename($obj), 0, -4)));
    }
    if (isset($_zp_page) && $_zp_page > 1) {
        echo '/' . $_zp_page;
    }
}

/**
 * Gets current item's owner (gallery images and albums) or author (Zenpage articles and pages)
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @global obj $_zp_current_album
 * @global obj $_zp_current_image
 * @global obj $_zp_current_zenpage_page
 * @global obj $_zp_current_zenpage_news
 * @param boolean $fullname If the owner/author has a real user account and there is a full name set it is returned
 * @return boolean
 */
function getOwnerAuthor($fullname = false) {
    global $_zp_current_album, $_zp_current_image, $_zp_current_zenpage_page, $_zp_current_zenpage_news;
    $ownerauthor = false;
    if (in_context(ZP_IMAGE)) {
        $ownerauthor = $_zp_current_image->getOwner($fullname);
    } else if (in_context(ZP_ALBUM)) {
        $ownerauthor = $_zp_current_album->getOwner($fullname);
    } 
    if (extensionEnabled('zenpage')) {
        if (is_Pages()) {
            $ownerauthor = $_zp_current_zenpage_page->getAuthor($fullname);
        } else if (is_NewsArticle()) {
            $ownerauthor = $_zp_current_zenpage_news->getAuthor($fullname);
        } 
    } 
    if ($ownerauthor) {
        return $ownerauthor;
    } 
    return false;
}

/**
 * Prints current item's owner (gallery images and albums) or author (Zenpage articles and pages)
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @param type $fullname
 */
function printOwnerAuthor($fullname = false) {
    echo html_encode(getOwnerAuthor($fullname));
}

/**
 * Returns the search url for items the current item's owner (gallery) or author (Zenpage) is assigned to
 * 
 * This eventually may return the url to an actual user profile page in the future.
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @return type
 */
function getOwnerAuthorURL() {
    $ownerauthor = getOwnerAuthor(false); 
    if($ownerauthor) {
        if (in_context(ZP_IMAGE) || in_context(ZP_ALBUM)) {
            return getUserURL($ownerauthor, 'gallery');
        } 
        if (extensionEnable('zenpagae') && (is_Pages() || is_NewsArticle())) {
            return getUserURL($ownerauthor, 'zenpage');
        } 
    }
}

/**
 * Prints the link to the search engine for results of all items the current item's owner (gallery) or author (Zenpage) is assigned to
 * 
 * This eventually may return the url to an actual user profile page in the future.
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @param type $fullname
 * @param type $resulttype
 * @param type $class
 * @param type $id
 * @param type $title
 */
function printOwnerAuthorURL($fullname = false, $resulttype = 'all', $class = null, $id = null, $title = null) {
    $author = $linktext = $title = getOwnerAuthor(false);
    if ($author) {
        if ($fullname) {
            $linktext = getOwnerAuthor(true);
        }
        if(is_null($title)) {
            $title = $linktext;
        }
        printUserURL($author, $resulttype, $linktext, $class, $id, $title);
    } 
}

/**
 * Returns a an url for the search engine for results of all items the user with $username is assigned to either as owner (gallery) or author (Zenpage)
 *  Note there is no check if the user name is actually a vaild user account name, owner or author! Use the *OwerAuthor() function for that instead
 * 
 * This eventually may return the url to an actual user profile page in the future.
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @param string $username The user name of a user. Note there is no check if the user name is actually valid!
 * @param string $resulttype  'all' for owner and author, 'gallery' for owner of images/albums only, 'zenpage' for author of news articles and pages
 * @return string|null
 */
function getUserURL($username, $resulttype = 'all') {
    if (empty($username)) {
        return null;
    }
    switch ($resulttype) {
        case 'all':
        default:
            $fields = array('owner', 'author');
            break;
        case 'gallery':
            $fields = array('owner');
            break;
        case 'zenpage':
            $fields = array('author');
            break;
    }
    return getSearchURL(search_quote($username), '', $fields, 1, null);
}

/**
 * Prints the link to the search engine for results of all items $username is assigned to either as owner (gallery) or author (Zenpage)
 * Note there is no check if the user name is actually a vaild user account name, owner or author! Use the *OwerAuthor() function for that instead
 * 
 * This eventually may point to an actual user profile page in the future.
 * 
 * @since ZenphotoCMS 1.5.2
 * 
 * @param string $username The user name of a user. 
 * @param string $resulttype  'all' for owner and author, 'gallery' for owner of images/albums only, 'zenpage' for author of news articles and pages
 * @param string $linktext The link text. If null the user name will be used
 * @param string $class The CSS class to attach, default null.
 * @param type $id The CSS id to attach, default null.
 * @param type $title The title attribute to attach, default null so the user name is used
 */
function printUserURL($username, $resulttype = 'all', $linktext = null, $class = null, $id = null, $title = null) {
    if ($username) {
        $url = getUserURL($username, $resulttype);
        if (is_null($linktext)) {
            $linktext = $username;
        }
        if (is_null($title)) {
            $title = $username;
        }
        printLinkHTML($url, $linktext, $title, $class, $id);
    }
}

/**
 * Display the site copyright notice if defined and display is enabled
 * 
 * @since ZenphotoCMS 1.5.8
 * 
 * @global obj $_zp_gallery
 * @param string $before Text to print before it
 * @param string $after Text to print after it
 */
function printCopyrightNotice($before = '', $after = '') {
    global $_zp_gallery;
    $copyright_notice = $_zp_gallery->getCopyrightNotice();
    if(!empty($copyright_notice) && getOption('display_copyright_notice')) {
        echo $before . $copyright_notice . $after;
    }
}

require_once(SERVERPATH . '/' . ZENFOLDER . '/template-filters.php');
?>
ZenphotoCMS 1.5.x API documentation generated by ApiGen