Skip to content

Instantly share code, notes, and snippets.

@simonwjackson
Last active October 27, 2017 18:21
Show Gist options
  • Save simonwjackson/89eda3190bef54a08d1e8a7b50a32752 to your computer and use it in GitHub Desktop.
Save simonwjackson/89eda3190bef54a08d1e8a7b50a32752 to your computer and use it in GitHub Desktop.
Tizen UI
.content-padding {
padding-top: 16px;
padding-bottom: 16px;
padding-left: 30px;
padding-right: 30px;
}
.button-group-height {
height: 72px;
}
/* ---------------------------------- LISTVIEW ---------------------------------- */
.ui-listview li .li-text-sub {
display: block;
text-overflow: ellipsis;
overflow: hidden;
line-height: 32px;
}
.ui-listview .li-text-sub-right-icon {
width: calc(100% - 34px);
}
.ui-listview li.li-has-2line {
padding-top: 18px;
padding-bottom: 17px;
height: 115px;
}
.ui-listview li.li-has-3line {
padding-top: 9px;
padding-bottom: 10px;
height: 128px;
}
.ui-listview.expand-list li.li-has-2line .ui-marquee{
-webkit-transform: translate(0, 16px);
}
.ui-listview.expand-list li.li-has-2line.ui-snap-listview-selected .ui-marquee{
-webkit-transform: translate3d(0, 0, 0);
-webkit-transition: all ease 1s;
margin-top: 0;
}
.ui-listview.expand-list li.li-has-2line .ui-li-sub-text{
-webkit-transform: translate(0, -20px);
opacity: 0;
}
.ui-listview.expand-list li.li-has-2line.ui-snap-listview-selected .ui-li-sub-text{
-webkit-transform: translate3d(0, 0, 0);
opacity: 1;
-webkit-transition: all ease 1s;
}
.ui-listview.expand-list li.li-has-3line .ui-marquee{
-webkit-transform: translate(0, 30px);
}
.ui-listview.expand-list li.li-has-3line.ui-snap-listview-selected .ui-marquee{
-webkit-transform: translate3d(0, 0, 0);
-webkit-transition: all ease 1s;
}
.ui-listview.expand-list li.li-has-3line .ui-li-sub-text:nth-child(2){
-webkit-transform: translate(0, -10px);
opacity: 0;
}
.ui-listview.expand-list li.li-has-3line.ui-snap-listview-selected .ui-li-sub-text:nth-child(2){
-webkit-transform: translate3d(0, 0, 0);
opacity: 1;
-webkit-transition: all ease 1s;
}
.ui-listview.expand-list li.li-has-3line .ui-li-sub-text:last-child{
-webkit-transform: translate(0, -42px);
opacity: 0;
}
.ui-listview.expand-list li.li-has-3line.ui-snap-listview-selected .ui-li-sub-text:last-child{
-webkit-transform: translate3d(0, 0, 0);
opacity: 1;
-webkit-transition: all ease 1s;
}
.ui-listview li.li-has-multiline {
padding-top: 6px;
padding-bottom: 5px;
min-height: 100px;
}
.ui-listview.ui-snap-listview li.li-has-multiline {
padding-top: 17px;
padding-bottom: 17px;
min-height: 108px;
line-height: normal;
}
.ui-listview li.li-has-radio.disabled,
.ui-listview li.li-has-radio.disabled > *,
.ui-listview li.li-has-checkbox.disabled,
.ui-listview li.li-has-checkbox.disabled > *,
.ui-listview li.li-has-multiline.disabled,
.ui-listview li.li-has-multiline.disabled > *,
.ui-listview li.li-has-multiline-sup.disabled,
.ui-listview li.li-has-multiline-sup.disabled > * {
color: rgb(51, 51, 51);
}
.ui-listview li.li-has-multiline > a {
margin-top: -6px;
margin-bottom: -6px;
padding-top: 11px;
padding-bottom: 12px;
}
.ui-listview li.li-has-multiline-sup {
padding-top: 6px;
padding-bottom: 5px;
min-height: 100px;
}
.ui-listview li.li-has-multiline-sup > a {
margin-top: -36px;
margin-bottom: -5px;
padding-top: 36px;
padding-bottom: 5px;
}
.ui-listview li {
position: relative;
}
.ui-listview li input[type="checkbox"]:not(.ui-switch-input),
.ui-listview li input[type="radio"] {
position: absolute;
right: 30px;
top: 50%;
-webkit-transform: translate3d(0, -50%, 0);
}
.ui-listview .li-has-toggle .ui-toggleswitch {
position: absolute;
right: 30px;
top: 50%;
-webkit-transform: translate3d(0, -50%, 0);
}
.ui-listview .li-has-radio.disabled,
.ui-listview li.li-has-multiline.li-has-radio.disabled,
.ui-listview li.li-has-multiline.li-has-radio.disabled .li-text-sub,
.ui-listview li.li-has-multiline.li-has-checkbox.disabled,
.ui-listview li.li-has-multiline.li-has-checkbox.disabled .li-text-sub {
color: rgb(51, 51, 51);
}
.ui-listview li.li-has-checkbox label,
.ui-listview li.li-has-radio label,
.ui-listview li.li-has-toggle > label {
display: block;
padding: 31.5px 94px 31.5px 40px;
margin-top: -30px;
margin-bottom: -21px;
margin-left: -38px;
margin-right: -8px;
}
.ui-listview li.li-has-thumb-left {
padding-left: 88px;
}
.ui-listview li.li-has-thumb-right {
padding-right: 88px;
}
.ui-listview.ui-snap-listview li.li-has-checkbox label,
.ui-listview.ui-snap-listview li.li-has-radio label,
.ui-listview.ui-snap-listview li.li-has-toggle > label,
.ui-listview.ui-snap-listview li.li-has-thumb-right{
padding: 27px 94px 21px 40px;
}
.ui-listview.ui-snap-listview li.li-has-thumb-left{
padding: 27px 40px 21px 102px;
}
.ui-listview li img {
position: absolute;
top: 50%;
-webkit-transform: translate3d(0, -50%, 0);
width: 56px;
height: 56px;
background-color: rgba(245, 245, 245, 1);
}
.ui-listview li.li-has-thumb-right img,
.ui-listview li.li-has-thumb-left img {
border-radius: 50%;
}
.ui-listview li.li-has-thumb-right img {
right: 16px;
}
.ui-listview li.li-has-thumb-left img {
left: 16px;
}
.ui-listview li.li-has-multiline.li-has-checkbox label,
.ui-listview li.li-has-multiline.li-has-radio label,
.ui-listview li.li-has-multiline.li-has-toggle label:not(.ui-toggleswitch) {
display: block;
padding-bottom: 12px;
padding-left: 16px;
padding-right: 90px;
padding-top: 11px;
margin-top: -6px;
margin-bottom: -6px;
margin-left: -16px;
margin-right: -16px;
}
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-checkbox label,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-radio label,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-toggle label:not(.ui-toggleswitch) {
padding-right: 104px;
}
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-checkbox label,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-radio label,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-toggle label:not(.ui-toggleswitch),
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-thumb-right,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-thumb-left {
padding-top: 27px;
}
.ui-listview li.li-has-multiline.li-has-thumb-left {
padding-left: 88px;
padding-right: 30px;
margin-left: 0;
}
.ui-listview li.li-has-multiline.li-has-toggle label:not(.ui-toggleswitch) {
padding-right: 75px;
}
.select-btn {
position: fixed;
top: 22px;
left: 50%;
margin-left: -50px;
height: 75px;
width: 100px;
line-height: 75px;
border-radius: 40px;
text-align: center;
background-color: rgb(8, 56, 94);
box-shadow: 0 3px 4px rgb(17, 17, 17);
z-index: 90;
}
.select-btn-text {
display: block;
height: 100%;
line-height: 60px;
background: url(../contents/list/images/tw_list_number_icon.png) no-repeat 50% 100%;
}
.ui-listview li.select {
background-color: rgb(8, 42, 57);
}
li.static::before {
content:"";
-webkit-mask-image: url(../contents/list/images/tw_list_check_ic.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(112, 186, 15, 1);
width: 0;
height: 56px;
position: absolute;
left: 40%;
-webkit-transform: translateX(-50%);
-webkit-transition: width 150ms, left 150ms;
}
.ui-listview-li li img {
height: 56px;
width: 56px;
background-color: rgb(8, 42, 57);
}
/* ------------------------------------------------------------------------------ */
/* ----------------------------------- NEXT DEPTH------------------------------------ */
.ui-listview li.li-has-next-depth {
padding-right: 60px;
}
.ui-listview li.li-has-next-depth a,
.ui-listview li.li-has-next-depth.disabled {
position: relative;
margin-right: -60px;
padding-right: 60px;
}
.ui-listview li.li-has-next-depth a:after {
position: absolute;
content: "";
right: 30px;
width: 50px;
height: 50px;
top: 50%;
margin-top: -25px;
background-repeat: no-repeat;
background-size: 50px 50px;
background-position: right center;
background-image: url(../extra/images/Controller_icon/tw_btn_next_depth_holo_dark.png);
}
.ui-listview li.li-has-next-depth.disabled a:after {
position: absolute;
right: 30px;
content: "";
width: 50px;
height: 50px;
top: 50%;
margin-top: -25px;
background-repeat: no-repeat;
background-size: 50px 50px;
background-position: right center;
background-image: url(../extra/images/Controller_icon/tw_btn_next_depth_disabled_holo_dark.png);
}
.ui-listview li.li-has-next-depth a:active:after,
.ui-listview li.li-has-next-depth a:focus:after {
position: absolute;
content: "";
right: 30px;
width: 50px;
height: 50px;
top: 50%;
margin-top: -25px;
background-repeat: no-repeat;
background-size: 50px 50px;
background-position: right center;
background-image: url(../extra/images/Controller_icon/tw_btn_next_depth_focused_holo_dark.png);
}
.ui-listview li.li-has-next-depth.disabled,
.ui-listview li.li-has-next-depth.disabled a,
.ui-listview li.li-has-next-depth.disabled .li-text-sub {
color: rgb(51, 51, 51);
}
.disabled {
pointer-events: none;
}
/* ------------------------------------------------------------------------------ */
/* -------------------------Action Icon------------------------------- */
.li-has-action-icon .li-action-text {
width: calc(100% - 88px);
height: 100%;
}
.li-has-action-icon .li-action-text::after {
content: "";
position: absolute;
height: 56px;
width: 2px;
background-color: rgba(245, 245, 245, 1);
right: 87px;
top: 50%;
transform: translate3d(0, -50%, 0);
}
.li-has-action-icon .li-action-icon-button {
position: absolute;
height: 100%;
width: 68px;
right: 0;
top: 0;
-webkit-mask-size: 53px 53px;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: center center;
padding: 0;
margin: 0;
margin-right: 20px;
}
.li-has-action-icon .li-action-icon-button.li-action-delete {
-webkit-mask-image: url(../contents/list/images/tw_list_delete_holo_dark.png);
background-color: rgba(245, 245, 245, 1);
}
.li-has-action-icon .li-action-icon-button.li-action-setting {
-webkit-mask-image: url(../contents/list/images/tw_list_setting_holo_dark.png);
background-color: rgba(245, 245, 245, 1);
}
.li-has-action-icon .li-action-icon-button.li-action-add {
-webkit-mask-image: url(../contents/list/images/tw_list_add_holo_dark.png);
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .li-has-action-icon.ui-li-active {
background-color: transparent;
}
.ui-listview .li-has-action-icon.ui-li-active .li-action-text:active {
background-color: rgba(8, 56, 94, 1);
}
.ui-listview .li-has-action-icon .li-action-icon-button:active {
background-color: rgba(8, 56, 94, 1);
}
/* ------------------------------------------------------------------------------ */
/*Basic Text*/
.basic-text-s-title {
margin:0 42px 60px 42px;
line-height: 40px;
font-size: 30px;
}
.basic-text-m-title {
margin:0 42px 60px 42px;
line-height: 44px;
font-size: 34px;
}
.basic-text-l-title {
margin:0 42px 60px 42px;
line-height: 48px;
font-size: 38px;
}
.basic-text-notitle{
margin: 60px 42px;
line-height: 48px;
font-size: 38px;
}
/* ------------------------------------------------------------------------------ */
/*Select List*/
#selectModePage #handler {
width:70px;
height: 70px;
position: fixed;
top: 50%;
transform:translate(0, -50%);
right: 0;
color: transparent;
display: none;
background-color: #a4a4a4;
-webkit-mask-image: url(../contents/list/images/b_drawer.png);
-webkit-mask-size: 18px 36px;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: 45px 19px;
z-index: 1;
}
#selectModePage li.select::after {
content:"";
-webkit-mask-image: url(../contents/list/images/tw_list_check_ic.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(115, 191, 15, 1);
width: 56px;
height: 56px;
position: absolute;
left: 50%;
-webkit-transform: translateX(-50%);
-webkit-transition: width 150ms, left 150ms;
}
#selectModePage #rightDrawer {
background-color: transparent;
}
#selectModePage #rightDrawer section {
height:360px !important;
}
#selectModePage .show-btn .select-btn {
visibility: visible;
transform: scale(1.0);
}
#selectModePage .select-btn {
pointer-events: auto;
visibility: hidden;
transition: transform 0.3s;
transform: scale(0);
}
#selectModePage .select-mode {
width: 100%;
height: 100%;
top: 0;
position: fixed;
pointer-events: none;
z-index:100;
}
#selectModePage .select-mode.open {
pointer-events: auto;
}
#selectModePage .select-mode.open::before {
display: block;
}
#selectModePage .select-mode::before {
content: "";
display: none;
z-index: 100;
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;
background-color: rgba(8,8,8,.7);
}
/*arrow*/
#selectModePage .select-mode::after {
transform: scale(0);
transition-property: transform;
transition-duration: 0s;
z-index: 100;
content: "";
position: fixed;
top: 48px;
width: 0;
height: 0;
border: 20px solid transparent;
border-bottom: 20px solid rgba(2, 56, 92, 1);
left: 50%;
margin-left: -20px;
}
#selectModePage .select-mode.open::after {
transform: scale(1.0);
transition-duration: 0.3s;
transform-origin : 50% 100%;
}
#selectModePage .select-mode.open .select-popup {
transform: scale(1.0);
z-index: 101;
visibility: visible;
pointer-events: auto;
}
#selectModePage .select-popup {
visibility: hidden;
pointer-events: none;
position: fixed;
left: 50%;
top: 83px;
margin-left: -100.5px;
width: 201px;
height: 201px;
transform: scale(0);
transform-origin : 50% 0;
transition: transform 0.3s;
padding: 0;
-webkit-border-radius: 50%;
-webkit-mask-image: -webkit-radial-gradient(#000000 6.28125rem, transparent 0);
min-height: 12.5625rem !important;
background-color: rgba(2, 56, 92, 1);
border: 0;
}
#selectModePage .select-popup .ui-listview {
margin: 0;
}
#selectModePage .select-popup .ui-listview::before {
display: none;
}
#selectModePage .select-popup .ui-listview::after {
display: none;
}
#selectModePage .select-popup .ui-listview li {
font-size: 1.75rem;
text-align: center;
border-bottom: 2px solid rgba(8, 8, 8, 1);
color: rgba(245, 245, 245, 1);
min-height: 100px;
border-image:none;
-webkit-mask-image: none;
margin-top: 0;
padding: 0;
}
#selectModePage .select-popup .ui-listview li.ui-li-active::before {
content: none;
}
#selectModePage .select-popup .ui-listview li:last-child {
border: 0;
}
#selectModePage .select-popup .ui-listview li a {
color: rgba(245, 245, 245, 1);
margin-top: 0;
line-height: 37px;
}
#selectModePage .select-popup .ui-listview li a:active {
color: rgba(245, 245, 245, 1);
background-color: rgba(23, 73, 115, 1);
}
#select-all {
padding-top: 45px;
padding-bottom: 19px;
}
#deselect-all {
padding-top: 19px;
padding-bottom: 44px;
}
/* -------------------------Animation List------------------------ */
#pageAnimation::after {
content: "This component is not supported in rectangular device";
position: absolute;
top: 0;
left: 0;
height: 100%;
display: -webkit-flex;
background-color: rgba(0, 0, 0, 0.6);
color: rgba(0,149,255,1);
text-align: center;
padding-left: 20px;
padding-right: 20px;
-webkit-align-items: center;
-webkit-justify-content: center;
}
/* -------------------------Marquee List------------------------ */
#pageMarqueeList::after {
content: "This component is not supported in rectangular device";
position: absolute;
top: 0;
left: 0;
height: 100%;
display: -webkit-flex;
background-color: rgba(0, 0, 0, 0.6);
color: rgba(0,149,255,1);
text-align: center;
padding-left: 20px;
padding-right: 20px;
-webkit-align-items: center;
-webkit-justify-content: center;
}
/* ----------------------------------- OPTION POPUP------------------------------------ */
.option-icon-action::before {
-webkit-mask-image:url(../contents/popup/images/b_option_list_icon_action.png);
}
.option-icon-delete::before {
-webkit-mask-image:url(../contents/popup/images/b_option_list_icon_delete.png);
}
.option-icon-enter::before {
-webkit-mask-image:url(../contents/popup/images/b_option_list_icon_enter.png);
}
.option-icon-share::before {
-webkit-mask-image:url(../contents/popup/images/b_option_list_icon_share.png);
}
.option-icon-notext-call::before {
-webkit-mask-image:url(../contents/popup/images/b_option_icon_call.png);
}
.option-icon-notext-contacts::before {
-webkit-mask-image:url(../contents/popup/images/b_option_icon_contacts.png);
}
.option-icon-notext-message::before {
-webkit-mask-image:url(../contents/popup/images/b_option_icon_message.png);
}
/* -----------------------------------POPUP------------------------------------ */
.btn-icon-share::before {
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url(../contents/popup/images/tw_ic_button_share_holo_dark.png);
}
.btn-icon-showonphone::before {
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url(../contents/popup/images/tw_ic_button_showonphone_holo_dark.png);
}
.btn-icon-delete::before {
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url(../contents/popup/images/tw_ic_button_delete_holo_dark.png);
}
.btn-icon-check::before {
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url(../contents/popup/images/tw_ic_popup_btn_check.png);
}
.btn-icon-cancel::before {
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url(../contents/popup/images/tw_ic_popup_btn_delete.png);
}
.btn-icon-toast-check::after {
background-color: rgba(255, 255, 255, 1);
-webkit-mask-image: url(../contents/popup/images/tw_ic_popup_btn_check.png);
}
/* -----------------------------------Content------------------------------------ */
.content-input-box {
display: block;
height: 82px;
line-height: 82px;
margin-top: 16px;
padding-left: 75px;
position: relative;
vertical-align: middle;
}
.content-input-box > input[type="checkbox"] {
position: absolute;
left: 15px;
top: 50%;
transform: translate3d(0, -50%, 0);
}
.content-input-box.multi-line {
display: block;
padding-left: 75px;
margin-top: 16px;
line-height: inherit;
}
.content-input-box-sub-text {
display: block;
}
/* ----------------------------------Toggle Switch-------------------------------- */
.ui-toggleswitch {
display: block;
margin: 0 auto;
}
/* -------------------------------------Controls---------------------------------- */
.checkbox-text,
.radio-text {
margin-top: 60px;
margin-bottom: 48px;
font-size: 34px;
text-align: center;
}
.checkbox-sub-text,
.radio-sub-text {
margin-top: 48px;
font-size: 26px;
text-align: center;
}
input[type="checkbox"]#checkboxControl {
display: block;
left: 50%;
transform: translate3d(-50%, 0, 0);
}
.radio-sample-container {
width: 100%;
text-align: center;
}
/* --------------------------------Processing------------------------------ */
.small-processing-container {
position: absolute;
top: 50%;
left: 50%;
-webkit-transform: translate3d(-50%, -50%, 0);
}
/* -------------------------Full Processing------------------------ */
#processingPage::after {
content: "This component is not supported in rectangular device";
position: absolute;
top: 0;
left: 0;
height: 100%;
display: -webkit-flex;
background-color: rgba(0, 0, 0, 0.6);
color: rgba(0,149,255,1);
text-align: center;
padding-left: 20px;
padding-right: 20px;
-webkit-align-items: center;
-webkit-justify-content: center;
}
/* --------------------------------Progress------------------------------ */
#pageSmallCircleProgressBar .result {
font-size:52px;
text-align:center;
display:inline;
line-height:56px;
margin-right:15px;
}
#pageSmallCircleProgressBar .ui-content{
display: flex;
display: -webkit-flex;
-webkit-align-items:center;
-webkit-justify-content: center;
}
#pageCircleProgressBar .ui-progressbar-large {
position: absolute;
top: 50%;
left: 50%;
-webkit-transform: translate(-50%, -50%);
}
#pageCircleProgressBar .result {
font-size:25px;
text-align:center;
top: 50%;
left: 50%;
-webkit-transform: translate(-50%, -50%);
position: absolute;
}
/* --------------------------------Rotary Event------------------------------ */
#pageRotaryEvent #circleprogress-container {
position: absolute;
top: 50%;
left: 50%;
transform: translate3d(-50%, -50%, 0);
}
#pageRotaryEvent #result {
font-size: 30px;
bottom: 50px;
position: fixed;
left: 50%;
transform: translate3d(-50%, 0, 0);
}
#pageRotaryEvent::after {
content: "This component is not supported in rectangular device";
position: absolute;
top: 0;
left: 0;
height: 100%;
display: -webkit-flex;
background-color: rgba(0, 0, 0, 0.6);
color: rgba(0,149,255,1);
text-align: center;
padding-left: 20px;
padding-right: 20px;
-webkit-align-items: center;
-webkit-justify-content: center;
}
/* -------------------------Circular Page Indicator------------------------ */
#pageIndicatorCirclePage::after {
content: "This component is not supported in rectangular device";
position: absolute;
top: 0;
left: 0;
height: 100%;
display: -webkit-flex;
background-color: rgba(0, 0, 0, 0.6);
color: rgba(0,149,255,1);
text-align: center;
padding-left: 20px;
padding-right: 20px;
-webkit-align-items: center;
-webkit-justify-content: center;
}
@media all and (-tizen-geometric-shape: circle) {
/* ------------------ LISTVIEW ------------------------ */
.ui-listview li img {
width: 76px;
height: 76px;
}
.ui-listview li{
-webkit-mask-image : -webkit-linear-gradient(left,transparent,#fff 30%,#fff 70%,transparent);
}
.ui-listview li.li-has-multiline > a {
margin-top: -17px;
margin-bottom: -17px;
padding-top: 17px;
padding-bottom: 17px;
}
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-2line-sub {
padding-top: 12px;
padding-bottom: 12px;
}
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-2line-sub > a {
margin-top: -12px;
margin-bottom: -12px;
padding-top: 12px;
padding-bottom: 12px;
}
.ui-listview li.li-has-thumb-right,
.ui-listview li.li-has-thumb-left,
.ui-listview li.li-has-action-icon,
.ui-listview li.li-has-toggle,
.ui-listview li.li-has-checkbox,
.ui-listview li.li-has-radio,
.ui-listview li.ui-li-btn
{
-webkit-mask-image: none;
}
.ui-listview li.li-has-checkbox label,
.ui-listview li.li-has-radio label,
.ui-listview li.li-has-toggle > label {
margin-top: -17px;
margin-bottom: -17px;
margin-left: -40px;
margin-right: -40px;
}
.ui-listview.ui-snap-listview li.li-has-checkbox label,
.ui-listview.ui-snap-listview li.li-has-radio label,
.ui-listview.ui-snap-listview li.li-has-toggle > label,
.ui-listview.ui-snap-listview li.li-has-thumb-right {
padding: 17px 100px 17px 30px;
}
.ui-listview .li-has-toggle .ui-toggleswitch {
right: 16px;
}
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-checkbox label,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-radio label,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-toggle label:not(.ui-toggleswitch) {
display: block;
padding: 17px 100px 17px 30px;
margin: -17px -40px -17px -40px;
}
.ui-listview.ui-snap-listview li.li-has-thumb-left {
padding: 17px 30px 17px 100px;
}
.ui-listview.ui-snap-listview li input[type="checkbox"]:not(.ui-switch-input),
.ui-listview.ui-snap-listview li input[type="radio"] {
right: 16px;
}
.li-has-action-icon .li-action-text::after{
height: 60px;
background-color: rgba(8, 8, 8, 0.3);
right: 103px;
}
.li-has-action-icon .li-action-icon-button{
width: 86px;
margin-left:2px;
margin-right: 15px;
}
.ui-listview .li-has-action-icon.ui-li-active .li-action-text:active {
background-color: transparent;
}
.ui-listview li.select {
background-color: transparent;
}
.ui-listview li.ui-listview-divider{
border-image: none;
}
.ui-listview li.ui-listview-divider:first-child {
margin-top: 0;
}
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-thumb-right,
.ui-listview.ui-snap-listview li.li-has-multiline.li-has-thumb-left {
padding-top: 17px;
}
.ui-listview.expand-list li.li-has-2line .ui-marquee{
-webkit-transform: translate(0, 5px);
}
.ui-listview.expand-list li.li-has-2line.ui-snap-listview-selected {
padding-top: 15px;
}
.ui-listview.expand-list li.li-has-2line.ui-snap-listview-selected .ui-marquee {
line-height: 47px;
}
/* ------------------ Animation List --------------------- */
#pageAnimation::after {
content: none;
}
/* ------------------- Marquee List --------------------- */
#pageMarqueeList::after {
content: none;
}
/* ------------------- Divide List --------------------- */
.li-has-action-icon .li-action-text::after {
content: none;
}
.li-has-action-icon .li-action-icon-button.li-action-delete,
.li-has-action-icon .li-action-icon-button.li-action-setting,
.li-has-action-icon .li-action-icon-button.li-action-add {
-webkit-mask-image: none;
background-color: transparent;
}
.li-has-action-icon .li-action-icon-button.li-action-delete::after,
.li-has-action-icon .li-action-icon-button.li-action-setting::after,
.li-has-action-icon .li-action-icon-button.li-action-add:after {
content: "";
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-mask-size: 53px 53px;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: center center;
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .li-has-action-icon {
overflow: visible;
}
.li-has-action-icon .li-action-icon-button.li-action-delete::after {
-webkit-mask-image: url(../contents/list/images/tw_list_delete_holo_dark.png);
}
.li-has-action-icon .li-action-icon-button.li-action-setting::after {
-webkit-mask-image: url(../contents/list/images/tw_list_setting_holo_dark.png);
}
.li-has-action-icon .li-action-icon-button.li-action-add::after {
-webkit-mask-image: url(../contents/list/images/tw_list_add_holo_dark.png);
}
.li-has-action-icon .li-action-icon-button {
position: absolute;
height: 100%;
width: 108px;
right: 0;
top: 0;
padding: 0;
margin: 0;
border: 0;
background-color: transparent;
overflow: visible;
}
.ui-listview .li-has-action-icon .li-action-icon-button:active {
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .li-has-action-icon .li-action-icon-button::before {
content: "";
position: absolute;
width: 108px;
height: 100%;
left:0;
top: 0;
-webkit-transform: scale(1);
transition: -webkit-transform 200ms linear;
border-radius: 50%;
background-color: rgba(30,30,30,0.7);
opacity: 0;
}
.ui-listview .li-has-action-icon .li-action-icon-button:active {
background-color: transparent;
}
.ui-listview .li-has-action-icon .li-action-icon-button:active::before {
opacity: 1;
-webkit-transform: scale(1.2);
}
/* ------------------ POPUP ------------------------ */
.ui-popup-content .btn-additional {
width: 160px;
height: 160px;
border-radius: 50% 50%;
background-color: rgba(10, 64, 94, 1);
margin: 19px auto auto;
padding: 59px 5px;
}
/* -------------Progress sample page-----------*/
#pageCircleProgressBar .ui-footer {
display: none;
}
#pageCircleProgressBar .result {
font-size:35px;
}
/* -------------Small progress sample page-----------*/
#pageSmallCircleProgressBar .ui-footer {
display: none;
}
#pageSmallCircleProgressBar .result{
-webkit-transform: translate(0, -70%);
}
#pageSmallCircleProgressBar .ui-progressbar-small{
-webkit-transform: translate(0, -70%);
}
/* -------------Marquee-----------*/
.ui-btn#start {
padding-left: 20%;
padding-right: 0;
}
.ui-btn#reset {
padding-right: 20%;
padding-left: 0;
}
/* -------------Rotary Event-----------*/
#pageRotaryEvent::after {
content: none;
}
/* -------------Circular Page Indicator-----------*/
#pageIndicatorCirclePage::after {
content: none;
}
/* -------------Full Processing-----------*/
#processingPage::after {
content: none;
}
#processingPage .ui-scroller::-webkit-scrollbar {
display: none;
}
/* -------------More Options-----------*/
#moreoptionsPopupCircle {
background-color: rgba(255, 255, 255, 0.1);
-webkit-mask-image: -webkit-radial-gradient(black 71%,transparent 0);
background-image: none;
}
}
.ui-item.ui-item-active::before{
background-color: rgba(2, 129, 209, 1);
}
.delete-icon::before {
position: absolute;
content: "";
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-mask-image: url(./images/b_more_option_delete_icon.png);
-webkit-mask-size: 100% 100%;
background-color: rgb(8, 8, 8);
-webkit-mask-repeat: no-repeat;
}
.human-icon::before {
position: absolute;
content: "";
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-mask-image: url(./images/b_more_option_human_icon.png);
-webkit-mask-size: 100% 100%;
background-color: rgb(8, 8, 8);
-webkit-mask-repeat: no-repeat;
}
.show-icon::before {
position: absolute;
content: "";
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-mask-image: url(./images/b_more_option_show_on_device_icon.png);
-webkit-mask-size: 100% 100%;
background-color: rgb(8, 8, 8);
-webkit-mask-repeat: no-repeat;
}
.x-icon::before {
position: absolute;
content: "";
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-mask-image: url(./images/tw_toast_popup_delete.png);
-webkit-mask-size: 100% 100%;
background-color: rgb(8, 8, 8);
-webkit-mask-repeat: no-repeat;
}
.fail-icon::before {
position: absolute;
content: "";
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-mask-image: url(./images/tw_toast_popup_fail.png);
-webkit-mask-size: 100% 100%;
background-color: rgb(8, 8, 8);
-webkit-mask-repeat: no-repeat;
}
/***************************************************************************
Body
***************************************************************************/
/***************************************************************************
Action Bar
***************************************************************************/
/****************************************************************************
Circle Progress Bar
****************************************************************************/
/****************************************************************************
Progress Bar
****************************************************************************/
/****************************************************************************
Processing
****************************************************************************/
/***************************************************************************
Buttons
***************************************************************************/
/***************************************************************************
Popup
***************************************************************************/
/***************************************************************************
Footer
***************************************************************************/
/***************************************************************************
Toast Popup
***************************************************************************/
/***************************************************************************
Option Popup
***************************************************************************/
/***************************************************************************
circular index scrollbar
***************************************************************************/
/***************************************************************************
Listview
***************************************************************************/
/****************************************************************************
Listview with Marquee
****************************************************************************/
/***************************************************************************
Checkbox/Radio
***************************************************************************/
/***************************************************************************
Page Indicator
***************************************************************************/
/****************************************************************************
Toggle Switch
****************************************************************************/
/***************************************************************************
Scroller
***************************************************************************/
/***************************************************************************
More options
***************************************************************************/
/************************************************************************/
/* fixedUIStyle.less */
/************************************************************************/
/* When system font-size is changed, winset size (ex. header, footer, button, etc,
include font) also changed. But in UX guide, some UI elements should have fixed
size regardless of system font size.
So, this less file supports fixed style for some elements.
- Layout: header, footer, button
- Popup : popup header, popup footer
- toggleswitch
- progress: text for ratio
*/
/***********************************************************************/
/* Layout(FIXED) */
/***********************************************************************/
/**********************************************************************/
/* Popup(FIXED) */
/**********************************************************************/
/**********************************************************************/
/* ToggleSwitch(FIXED) */
/**********************************************************************/
/**********************************************************************/
/* List(FIXED) */
/**********************************************************************/
/************************
Listview
*************************/
/***************************************************************************
Common Style
***************************************************************************/
/***************************************************************************
Development Tip
If you want to implement specific css code of circle device,
you should implement specific code here.
But when you implement code, you should check duplicate value
between common/layout.less and this file.
If you have duplicate value, please change static value to variable value.
***************************************************************************/
body {
background: rgba(8, 8, 8, 1);
color: rgba(199, 199, 199, 1);
}
.ui-header {
height: 115px;
line-height: 39px;
background-color: transparent;
}
.ui-header .ui-title {
color: rgba(18, 180, 255, 1);
text-align: center;
padding-top: 51px;
margin-left: 2.5rem;
margin-right: 2.5rem;
padding-left: 1.5rem;
padding-right: 1.5rem;
font-size: 30px;
-webkit-mask-image: -webkit-linear-gradient(left, #000000 88%, rgba(0, 0, 0, 0) 95%);
}
.ui-header .ui-title.ui-icon {
background: none !important;
padding-left: 1.5rem;
}
.ui-header.ui-header-small {
height: 76px;
}
.ui-header.ui-header-small .ui-title {
padding-top: 37px;
}
.ui-header.ui-has-more .ui-title {
padding-right: 1.5rem;
-webkit-mask-image: -webkit-linear-gradient(left, #000000 88%, rgba(0, 0, 0, 0) 95%);
}
.ui-header.ui-has-more .ui-more {
position: fixed;
top: 50%;
right: 0;
width: 34px;
height: 200px;
transform: translate(0, -50%);
}
.ui-header.ui-has-more .ui-more.ui-icon-overflow {
-webkit-mask-image: none;
-moz-mask-image: none;
-ms-mask-image: none;
-o-mask-image: none;
mask-image: none;
-webkit-mask-position: initial;
-moz-mask-position: initial;
-ms-mask-position: initial;
-o-mask-position: initial;
mask-position: initial;
background-color: transparent;
overflow: visible;
}
.ui-header.ui-has-more .ui-more.ui-icon-overflow[disabled="disabled"] {
-webkit-mask-position: initial;
-moz-mask-position: initial;
-ms-mask-position: initial;
-o-mask-position: initial;
mask-position: initial;
}
.ui-header.ui-has-more .ui-more.ui-icon-overflow:active {
background-color: transparent;
}
.ui-header.ui-has-more .ui-more::before {
content: "";
position: absolute;
width: 8.75rem;
height: 8.75rem;
background-color: rgba(60, 60, 60, 0.8);
border-radius: 50%;
-webkit-transform: scale(0);
-moz-transform: scale(0);
-ms-transform: scale(0);
-o-transform: scale(0);
transform: scale(0);
top: 50%;
left: 50%;
margin-top: -4.375rem;
margin-left: -4.375rem;
-webkit-transition: transform ease-out 300ms;
-moz-transition: transform ease-out 300ms;
-o-transition: transform ease-out 300ms;
-ms-transition: transform ease-out 300ms;
transition: transform ease-out 300ms;
}
.ui-header.ui-has-more .ui-more:active::before {
-webkit-transition: transform ease-out 300ms;
-moz-transition: transform ease-out 300ms;
-o-transition: transform ease-out 300ms;
-ms-transition: transform ease-out 300ms;
transition: transform ease-out 300ms;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
.ui-header.ui-has-more .ui-more::after {
content: "";
position: absolute;
width: 1rem;
height: 2.25rem;
-webkit-mask-box-image: url("images/Actionbar/b_more_option.png");
-moz-mask-box-image: url("images/Actionbar/b_more_option.png");
-ms-mask-box-image: url("images/Actionbar/b_more_option.png");
-o-mask-box-image: url("images/Actionbar/b_more_option.png");
mask-box-image: url("images/Actionbar/b_more_option.png");
background-color: rgba(245, 245, 245, 1);
border-radius: 50%;
top: 50%;
right: 0.375rem;
-webkit-transform: translate(0, -50%);
-moz-transform: translate(0, -50%);
-ms-transform: translate(0, -50%);
-o-transform: translate(0, -50%);
transform: translate(0, -50%);
}
.ui-page {
background: rgba(8, 8, 8, 1);
}
.ui-page .ui-content {
text-align: center;
font-size: 32px;
}
.ui-page .ui-footer.ui-bottom-button {
position: fixed !important;
bottom: 0;
z-index: 1000;
height: 28%;
}
.ui-page .ui-footer.ui-bottom-button .ui-btn {
left: 50%;
width: 100%;
height: 130%;
line-height: 2.625rem;
position: relative;
margin: 0;
padding: 8% 22% 16%;
background: -webkit-radial-gradient(center, circle cover, rgba(0, 60, 102, 0.9) 50%, transparent 50%);
background-size: 300% 840%;
background-position: 50% 16.5%;
background-repeat: no-repeat;
-webkit-transform: translate(-50%, 0);
}
.ui-page .ui-footer.ui-bottom-button .ui-btn::after {
content: "";
position: absolute;
width: 100%;
height: 100%;
left: 0;
top: 0;
background: -webkit-radial-gradient(center, circle cover, rgba(0, 149, 255, 0.5) 50%, transparent 50%);
background-position: 50% 16.5%;
background-repeat: no-repeat;
background-size: 300% 840%;
-webkit-transform: translate(0, 10%);
-webkit-transition: -webkit-transform 250ms;
opacity: 0;
z-index: -1;
}
.ui-page .ui-footer.ui-bottom-button .ui-btn:active::after {
opacity: 1;
-webkit-transform: translate(0, 0);
}
.ui-page .ui-footer.ui-bottom-button .ui-btn.ui-btn-icon-only::before {
top: 0;
-webkit-transform: translate(-50%, 56%);
-webkit-transition: transform 150ms ease-out;
}
/***************************************************************************
Page Scroll
***************************************************************************/
.ui-page.ui-scroll-on {
overflow: hidden;
-webkit-overflow-scrolling: auto;
-moz-overflow-scrolling: auto;
-o-overflow-scrolling: auto;
-ms-overflow-scrolling: auto;
overflow-scrolling: auto;
}
.ui-page.ui-scroll-on .ui-content {
margin-top: 4.75rem;
padding-bottom: 6.25rem;
padding-top: 0;
}
.ui-page.ui-scroll-on .ui-header ~ .ui-content {
margin-top: 0;
}
.ui-page.ui-scroll-on .ui-header ~ .ui-scroller > .ui-content {
margin-top: 0;
}
.ui-page.ui-scroll-on .ui-content ~ .ui-footer:not(.ui-expandable-footer) {
margin-top: -5.3125rem;
}
/*
* Common definition for theme
*/
/***************************************************************
default font size (base font from WRT)
=> small: 13px
=> normal: 17px
=> large: 20px
This is only applied to gear2.
so, we set @font_size_default as 17px on gear2 and it was changed to 16px since gear-S.
This value only used for @rem_base, not for html font-size
html font-size is set by WRT base font-size
***************************************************************/
a {
color: rgba(199, 199, 199, 1);
}
/***************************************************************************
toggle switch
***************************************************************************/
.ui-switch-text {
margin-top: 60px;
margin-bottom: 48px;
font-size: 34px;
padding: 0;
}
.ui-switch-sub-text {
margin-top: 48px;
font-size: 26px;
color: rgba(18, 180, 255, 1);
}
.ui-toggleswitch {
width: 76px;
height: 76px;
}
.ui-toggleswitch.ui-toggleswitch-large {
width: 76px;
height: 76px;
}
.ui-toggleswitch .ui-switch-input {
background-color: rgba(245, 245, 245, 0.3);
-webkit-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_bg_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_bg_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_bg_holo_dark.png');
-o-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_bg_holo_dark.png');
mask-image: url('images/Controller_icon/On_Off/tw_btn_control_bg_holo_dark.png');
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
-webkit-transition: all 300ms;
-moz-transition: all 300ms;
-o-transition: all 300ms;
-ms-transition: all 300ms;
transition: all 300ms;
}
.ui-toggleswitch .ui-switch-input::before {
content: none;
}
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-activation,
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-button {
background-color: transparent;
}
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-button::before {
background-color: transparent;
}
.ui-toggleswitch .ui-switch-activation,
.ui-toggleswitch .ui-switch-button {
width: 100%;
height: 100%;
-webkit-mask-image: none;
background-color: transparent;
transform: none;
}
.ui-toggleswitch .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-button::before {
-webkit-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_on_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_on_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_on_holo_dark.png');
-o-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_on_holo_dark.png');
mask-image: url('images/Controller_icon/On_Off/tw_btn_control_on_holo_dark.png');
-webkit-transform: scale(0);
-moz-transform: scale(0);
-ms-transform: scale(0);
-o-transform: scale(0);
transform: scale(0);
-webkit-transition: all 300ms;
-moz-transition: all 300ms;
-o-transition: all 300ms;
-ms-transition: all 300ms;
transition: all 300ms;
background-color: rgba(245, 245, 245, 1);
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
-webkit-mask-position: 0;
-moz-mask-position: 0;
-ms-mask-position: 0;
-o-mask-position: 0;
mask-position: 0;
}
.ui-toggleswitch .ui-switch-activation::after,
.ui-toggleswitch .ui-switch-button::after {
content: "";
width: 100%;
height: 100%;
-webkit-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_off_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_off_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_off_holo_dark.png');
-o-mask-image: url('images/Controller_icon/On_Off/tw_btn_control_off_holo_dark.png');
mask-image: url('images/Controller_icon/On_Off/tw_btn_control_off_holo_dark.png');
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
-webkit-mask-position: 0;
-moz-mask-position: 0;
-ms-mask-position: 0;
-o-mask-position: 0;
mask-position: 0;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
-webkit-transition: transform 300ms;
-moz-transition: transform 300ms;
-o-transition: transform 300ms;
-ms-transition: transform 300ms;
transition: transform 300ms;
background-color: rgba(245, 245, 245, 0.4);
position: absolute;
left: 0;
}
.ui-toggleswitch .ui-switch-input:checked {
background-color: rgba(112, 186, 15, 1);
}
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-activation,
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-button {
width: 100%;
height: 100%;
-webkit-mask-image: none;
background-color: transparent;
transform: none;
}
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-button::before {
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
background-color: rgba(245, 245, 245, 1);
-webkit-mask-position: 0;
-moz-mask-position: 0;
-ms-mask-position: 0;
-o-mask-position: 0;
mask-position: 0;
}
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-activation::after,
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-button::after {
-webkit-transform: scale(0);
-moz-transform: scale(0);
-ms-transform: scale(0);
-o-transform: scale(0);
transform: scale(0);
}
.ui-toggleswitch .ui-switch-input:checked:active::before {
background-color: rgba(112, 186, 15, 1);
}
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-activation,
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-button {
background-color: transparent;
}
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-button::before {
background-color: rgba(245, 245, 245, 1);
}
/***************************************************************************
popup
***************************************************************************/
.ui-popup {
width: 100%;
height: 100%;
overflow: hidden;
border: 0;
border-radius: 50% 50%;
background-image: -webkit-radial-gradient(transparent -100%, transparent -90%, rgba(8, 8, 8, 1) 69%), -webkit-radial-gradient(rgba(8, 8, 8, 1) 71%, transparent 0px);
}
.ui-popup ::-webkit-scrollbar {
display: none;
}
.ui-popup .ui-popup-header {
text-align: center;
min-height: 99px;
background-color: transparent;
color: rgba(18, 180, 255, 1);
padding-top: 60px;
padding-right: 61px;
padding-bottom: 0;
padding-left: 61px;
}
.ui-popup .ui-popup-header + .ui-popup-content {
padding-top: 26px;
}
.ui-popup .ui-popup-content {
color: rgba(199, 199, 199, 1);
background-color: transparent;
font-size: 32px;
padding: 16% 8% 4%;
text-align: center;
}
.ui-popup .ui-popup-content .ui-listview,
.ui-popup .ui-popup-content .ui-inline-listview {
margin: -1.5625rem -3.4375rem -1rem -3.4375rem;
}
.ui-popup.ui-has-side-buttons .ui-popup-content {
padding-left: 19%;
padding-right: 19%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn {
height: 100%;
width: 26%;
border: none;
position: absolute;
text-indent: -9999px;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn::before {
content: "";
-webkit-mask-size: 50px 50px;
-webkit-mask-repeat: no-repeat;
width: 50px;
height: 50px;
position: absolute;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn::after {
content: "";
position: absolute;
width: 100%;
height: 100%;
top: 0;
left: 0;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(n) {
top: 50%;
-webkit-transform: translate3d(68%, -50%, 0);
-moz-transform: translate3d(68%, -50%, 0);
-ms-transform: translate3d(68%, -50%, 0);
-o-transform: translate3d(68%, -50%, 0);
transform: translate3d(68%, -50%, 0);
background: -webkit-radial-gradient(center, circle cover, rgba(0, 60, 102, 0.9) 70%, transparent 70%);
background-repeat: no-repeat;
background-size: 570% 150%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(n)::before {
top: 50%;
-webkit-transform: translate3d(0, -50%, 0);
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(n)::after {
background: -webkit-radial-gradient(center, circle cover, rgba(0, 149, 255, 0.5) 70%, transparent 70%);
background-repeat: no-repeat;
background-size: 570% 150%;
-webkit-transition: -webkit-transform 250ms;
opacity: 0;
z-index: -1;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(n):active::after {
opacity: 1;
-webkit-transform: translate(0, 0);
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(1) {
-webkit-transform: translate3d(68%, -50%, 0);
-moz-transform: translate3d(68%, -50%, 0);
-ms-transform: translate3d(68%, -50%, 0);
-o-transform: translate3d(68%, -50%, 0);
transform: translate3d(68%, -50%, 0);
background-position: 100% 50%;
right: 100%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(1)::before {
left: 38%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(1)::after {
background-position: 100% 50%;
-webkit-transform: translate(-20%, 0);
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(1):active {
background-position: 100% 50%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(2) {
-webkit-transform: translate3d(-68%, -50%, 0);
-moz-transform: translate3d(-68%, -50%, 0);
-ms-transform: translate3d(-68%, -50%, 0);
-o-transform: translate3d(-68%, -50%, 0);
transform: translate3d(-68%, -50%, 0);
left: 100%;
background-position: 0 50%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(2)::before {
right: 38%;
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(2)::after {
background-position: 0 50%;
-webkit-transform: translate(20%, 0);
}
.ui-popup .ui-popup-footer.ui-side-button .ui-btn:nth-child(2):active {
background-position: 0 50%;
}
.ui-popup .ui-popup-footer.ui-bottom-button .ui-btn {
position: absolute;
top: 100%;
left: 50%;
height: 36%;
line-height: 2.625rem;
width: 100%;
border: none;
padding: 8% 22% 16%;
background: -webkit-radial-gradient(center, circle cover, rgba(0, 60, 102, 0.9) 50%, transparent 50%);
background-position: 50% 16.5%;
background-repeat: no-repeat;
background-size: 300% 840%;
-webkit-transform: translate3d(-50%, -77%, 0);
-moz-transform: translate3d(-50%, -77%, 0);
-ms-transform: translate3d(-50%, -77%, 0);
-o-transform: translate3d(-50%, -77%, 0);
transform: translate3d(-50%, -77%, 0);
}
.ui-popup .ui-popup-footer.ui-bottom-button .ui-btn.ui-btn-icon-only::before {
top: 1.625rem;
-webkit-transform: translate3d(-50%, 0, 0);
-moz-transform: translate3d(-50%, 0, 0);
-ms-transform: translate3d(-50%, 0, 0);
-o-transform: translate3d(-50%, 0, 0);
transform: translate3d(-50%, 0, 0);
}
.ui-popup .ui-popup-footer.ui-bottom-button .ui-btn::after {
content: "";
position: absolute;
width: 100%;
height: 100%;
left: 0;
top: 0;
background: -webkit-radial-gradient(center, circle cover, rgba(0, 149, 255, 0.5) 50%, transparent 50%);
background-position: 50% 16.5%;
background-repeat: no-repeat;
background-size: 300% 840%;
-webkit-transform: translate(0, 10%);
-webkit-transition: -webkit-transform 250ms;
opacity: 0;
z-index: -1;
}
.ui-popup .ui-popup-footer.ui-bottom-button .ui-btn:active::after {
opacity: 1;
-webkit-transform: translate(0, 0);
}
/***********************
Popup Container Scroll
***********************/
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-wrapper::after {
content: "";
position: absolute;
left: 0;
top: 0;
width: 100%;
height: 100%;
pointer-events: none;
background-color: rgba(18, 180, 255, 0.5);
-webkit-mask-image: url(./images/Popup/b_popup_edge_inner_glow.png);
-webkit-mask-size: 100% 100%;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-content {
margin-top: 4.75rem;
margin-bottom: 4.75rem;
padding-top: 0;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-header + .ui-popup-content {
margin-top: 0;
padding-top: 8px;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-footer {
margin-top: -4.75rem;
}
/****************************************
Popup Toast
****************************************/
@-webkit-keyframes toastCheck {
0% {
-webkit-mask-position: 0px 0px;
}
6% {
-webkit-mask-position: -82px 0px;
}
11% {
-webkit-mask-position: -164px 0px;
}
17% {
-webkit-mask-position: -246px 0px;
}
22% {
-webkit-mask-position: 0px -82px;
}
28% {
-webkit-mask-position: -82px -82px;
}
33% {
-webkit-mask-position: -164px -82px;
}
39% {
-webkit-mask-position: -246px -82px;
}
44% {
-webkit-mask-position: 0px -164px;
}
50% {
-webkit-mask-position: -82px -164px;
}
56% {
-webkit-mask-position: -164px -164px;
}
61% {
-webkit-mask-position: -246px -164px;
}
67% {
-webkit-mask-position: 0px -246px;
}
72% {
-webkit-mask-position: -82px -246px;
}
78% {
-webkit-mask-position: -164px -246px;
}
83% {
-webkit-mask-position: -246px -246px;
}
89% {
-webkit-mask-position: 0px -328px;
}
94% {
-webkit-mask-position: -82px -328px;
}
100% {
-webkit-mask-position: -164px -328px;
}
}
.ui-popup.ui-popup-toast {
min-height: 100%;
background-color: rgba(8, 8, 8, 1);
background-image: none;
border: none;
margin: 0;
}
.ui-popup.ui-popup-toast .ui-popup-wrapper {
height: 100vh !important;
}
.ui-popup.ui-popup-toast .ui-popup-content {
height: auto;
position: relative;
top: 50%;
transform: translateY(-50%);
background-color: transparent;
color: rgba(245, 245, 245, 1);
font-size: 32px;
text-align: center;
line-height: 43px;
}
.ui-popup.ui-popup-toast.ui-popup-toast-graphic .ui-popup-toast-check-icon {
width: 5.125rem;
height: 5.125rem;
-webkit-mask-image: url("./images/Popup/toast_check.png");
-webkit-mask-repeat: no-repeat;
-webkit-animation-name: toastCheck;
-webkit-animation-duration: 300ms;
-webkit-animation-delay: 300ms;
-webkit-animation-iteration-count: 1;
-webkit-animation-timing-function: step-start;
-webkit-animation-fill-mode: forwards;
background-color: rgba(224, 224, 224, 1);
}
.ui-popup.ui-popup-toast.ui-popup-toast-graphic .ui-popup-toast-check-icon::after {
content: none;
}
/***********************
Option Popup
***********************/
.ui-popup.ui-ctxpopup {
height: auto;
}
/***************************************************************************
listview
***************************************************************************/
.ui-listview {
padding-bottom: 1.25rem;
}
.ui-listview li {
margin-top: 0.6875rem;
font-size: 2.25rem;
min-height: 6.75rem;
padding: 1.09375rem 2.5rem;
border: 0 none;
border-bottom: 1.5px solid transparent;
border-image: -webkit-linear-gradient(0deg, transparent, #245b85, transparent) 1;
color: rgba(245, 245, 245, 1);
text-align: center;
line-height: 4.5625rem;
}
.ui-listview li.ui-listview-divider {
height: 2.0625rem;
min-height: 2.0625rem;
line-height: 1.9375rem;
background-color: transparent;
color: rgba(18, 180, 255, 1);
display: -webkit-flex;
padding: 0;
}
.ui-listview li.ui-listview-divider::before {
content: "";
height: 100%;
width: auto;
-webkit-mask-image: url("images/listview/tw_list_sub_line_left.png");
-moz-mask-image: url("images/listview/tw_list_sub_line_left.png");
-ms-mask-image: url("images/listview/tw_list_sub_line_left.png");
-o-mask-image: url("images/listview/tw_list_sub_line_left.png");
mask-image: url("images/listview/tw_list_sub_line_left.png");
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 2px;
-moz-mask-size: 100% 2px;
-ms-mask-size: 100% 2px;
-o-mask-size: 100% 2px;
mask-size: 100% 2px;
-webkit-mask-position: -0.75rem 50%;
-moz-mask-position: -0.75rem 50%;
-ms-mask-position: -0.75rem 50%;
-o-mask-position: -0.75rem 50%;
mask-position: -0.75rem 50%;
background: rgba(18, 180, 255, 0.28);
-webkit-flex: 1;
}
.ui-listview li.ui-listview-divider::after {
content: "";
height: 100%;
width: auto;
-webkit-mask-image: url("images/listview/tw_list_sub_line_right.png");
-moz-mask-image: url("images/listview/tw_list_sub_line_right.png");
-ms-mask-image: url("images/listview/tw_list_sub_line_right.png");
-o-mask-image: url("images/listview/tw_list_sub_line_right.png");
mask-image: url("images/listview/tw_list_sub_line_right.png");
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 2px;
-moz-mask-size: 100% 2px;
-ms-mask-size: 100% 2px;
-o-mask-size: 100% 2px;
mask-size: 100% 2px;
-webkit-mask-position: 0.75rem 50%;
-moz-mask-position: 0.75rem 50%;
-ms-mask-position: 0.75rem 50%;
-o-mask-position: 0.75rem 50%;
mask-position: 0.75rem 50%;
background: rgba(18, 180, 255, 0.28);
-webkit-flex: 1;
}
.ui-listview li.ui-li-btn {
padding: 0;
}
.ui-listview li.ui-li-btn .ui-btn {
padding: 2.625rem 2.5rem;
margin: 0;
line-height: inherit;
}
.ui-listview li.ui-li-btn.ui-grid-col-2 .ui-btn {
padding: 1.75rem 1.25rem;
}
.ui-listview li.ui-li-btn.ui-grid-col-3 .ui-btn {
padding: 1.75rem 1.875rem;
height: 100%;
text-indent: -9999px;
}
.ui-listview li > * {
color: rgba(245, 245, 245, 1);
}
.ui-listview li > a,
.ui-listview li button {
padding: 1.09375rem 2.5rem;
margin: -1.09375rem -2.5rem;
}
.ui-listview li.ui-li-active {
background-color: transparent;
color: rgba(245, 245, 245, 1);
}
.ui-listview li.ui-li-active::before {
content: "";
position: absolute;
width: 100%;
height: 100%;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
-webkit-mask-image: url("images/List/b_list_focus_bg_108.png");
-webkit-mask-size: 100% 100%;
background-color: rgba(245, 245, 245, 0.65);
pointer-events: none;
}
.ui-listview li:last-child {
border: 0;
}
.ui-listview.ui-snap-listview .ui-li-sub-text {
color: rgba(18, 180, 255, 1);
font-size: 1.5rem;
}
.ui-listview.ui-snap-listview .ui-snap-listview-selected::before {
content: "";
position: absolute;
width: 100%;
height: 100%;
left: 50%;
top: 50%;
-webkit-transform: translate(-50%, -50%);
-moz-transform: translate(-50%, -50%);
-ms-transform: translate(-50%, -50%);
-o-transform: translate(-50%, -50%);
transform: translate(-50%, -50%);
-webkit-mask-image: url("images/List/b_list_focus_bg_108.png");
-webkit-mask-size: 100% 100%;
background-color: rgba(245, 245, 245, 0.5);
border-radius: 20vw;
pointer-events: none;
}
.ui-listview.ui-snap-listview .ui-snap-listview-selected.ui-li-active {
color: rgba(245, 245, 245, 1);
}
.ui-listview.ui-snap-listview .ui-snap-listview-selected.ui-li-active::before {
background-color: rgba(245, 245, 245, 0.65);
}
.ui-listview.ui-snap-listview li {
border: none;
}
.ui-listview.ui-snap-listview li.ui-li-active::before {
background-color: transparent;
}
/***************************************************************************
listview extra
***************************************************************************/
.ui-listview .ui-li-has-action-icon .ui-action-text::after {
background-color: rgba(199, 199, 199, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-delete {
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-setting {
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-add {
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon.ui-li-active .ui-action-text:active {
background-color: rgba(8, 56, 94, 1);
color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-delete:active,
.ui-listview .ui-li-has-action-icon .ui-action-setting:active,
.ui-listview .ui-li-has-action-icon .ui-action-add:active {
background-color: rgba(8, 56, 94, 1);
color: rgba(245, 245, 245, 1);
}
.ui-listview .li-divider {
height: 2.0625rem;
min-height: 2.0625rem;
line-height: 1.9375rem;
background-color: rgba(18, 180, 255, 0.28);
color: rgba(18, 180, 255, 1);
padding: 0;
}
/***************************************************************************
checkbox / radio button
***************************************************************************/
input[type="checkbox"]:not(.ui-switch-input),
input[type="radio"] {
height: 76px;
width: 76px;
}
input[type="checkbox"]:not(.ui-switch-input) {
-webkit-mask-image: none;
background: -webkit-radial-gradient(rgba(245, 245, 245, 0.3) 67%, transparent 71%, transparent), url("images/Controller_icon/Check/ref_btn_check_holo_dark.png") no-repeat;
background-size: 100% 100%;
}
input[type="checkbox"]:not(.ui-switch-input)::before {
content: "";
-webkit-mask-image: none;
background: -webkit-radial-gradient(rgba(112, 186, 15, 1) 67%, transparent 71%, transparent);
height: 100%;
width: 100%;
opacity: 0;
position: absolute;
-webkit-transition: all 300ms ease-in;
-moz-transition: all 300ms ease-in;
-o-transition: all 300ms ease-in;
-ms-transition: all 300ms ease-in;
transition: all 300ms ease-in;
}
input[type="checkbox"]:not(.ui-switch-input)::after {
-webkit-mask-image: url('images/Controller_icon/Check/tw_btn_check_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/Check/tw_btn_check_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/Check/tw_btn_check_holo_dark.png');
-o-mask-image: url('images/Controller_icon/Check/tw_btn_check_holo_dark.png');
mask-image: url('images/Controller_icon/Check/tw_btn_check_holo_dark.png');
-webkit-mask-repeat: repeat;
-moz-mask-repeat: repeat;
-ms-mask-repeat: repeat;
-o-mask-repeat: repeat;
mask-repeat: repeat;
-webkit-mask-position: center bottom;
-moz-mask-position: center bottom;
-ms-mask-position: center bottom;
-o-mask-position: center bottom;
mask-position: center bottom;
background-color: rgba(245, 245, 245, 1);
height: 0;
-webkit-transition-property: height;
-moz-transition-property: height;
-o-transition-property: height;
-ms-transition-property: height;
transition-property: height;
-webkit-transition-duration: 500ms;
-moz-transition-duration: 500ms;
-o-transition-duration: 500ms;
-ms-transition-duration: 500ms;
transition-duration: 500ms;
bottom: 0;
-webkit-mask-size: 100%;
-moz-mask-size: 100%;
-ms-mask-size: 100%;
-o-mask-size: 100%;
mask-size: 100%;
}
input[type="checkbox"]:not(.ui-switch-input):checked::before {
-webkit-animation: checkon 300ms ease-out;
-moz-animation: checkon 300ms ease-out;
-ms-animation: checkon 300ms ease-out;
-o-animation: checkon 300ms ease-out;
animation: checkon 300ms ease-out;
-webkit-transition: none;
-moz-transition: none;
-o-transition: none;
-ms-transition: none;
transition: none;
-webkit-mask-image: none;
background: -webkit-radial-gradient(rgba(112, 186, 15, 1) 67%, transparent 71%, transparent);
opacity: 1;
}
input[type="checkbox"]:not(.ui-switch-input):checked::after {
height: 100%;
}
input[type="checkbox"]:not(.ui-switch-input):active {
background-color: transparent;
}
input[type="checkbox"]:not(.ui-switch-input):active::after {
width: 100%;
height: 0;
}
input[type="checkbox"]:not(.ui-switch-input):active:checked::before {
-webkit-mask-image: none;
background: -webkit-radial-gradient(rgba(112, 186, 15, 1) 67%, transparent 71%, transparent);
}
input[type="checkbox"]:not(.ui-switch-input):active:checked::after {
background-color: rgba(245, 245, 245, 1);
height: 100%;
}
@-webkit-keyframes checkon {
from {
opacity: 0.5;
-webkit-transform: scale(0.8);
-moz-transform: scale(0.8);
-ms-transform: scale(0.8);
-o-transform: scale(0.8);
transform: scale(0.8);
}
100% {
opacity: 1.0;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
}
@-o-keyframes slideupinreverse {
from {
opacity: 0.5;
-webkit-transform: scale(0.8);
-moz-transform: scale(0.8);
-ms-transform: scale(0.8);
-o-transform: scale(0.8);
transform: scale(0.8);
}
100% {
opacity: 1.0;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
}
@-ms-keyframes slideupinreverse {
from {
opacity: 0.5;
-webkit-transform: scale(0.8);
-moz-transform: scale(0.8);
-ms-transform: scale(0.8);
-o-transform: scale(0.8);
transform: scale(0.8);
}
100% {
opacity: 1.0;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
}
@-moz-keyframes slideupinreverse {
from {
opacity: 0.5;
-webkit-transform: scale(0.8);
-moz-transform: scale(0.8);
-ms-transform: scale(0.8);
-o-transform: scale(0.8);
transform: scale(0.8);
}
100% {
opacity: 1.0;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
}
@keyframes slideupinreverse {
from {
opacity: 0.5;
-webkit-transform: scale(0.8);
-moz-transform: scale(0.8);
-ms-transform: scale(0.8);
-o-transform: scale(0.8);
transform: scale(0.8);
}
100% {
opacity: 1.0;
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
}
input[type="radio"] {
background: -webkit-radial-gradient(transparent, transparent 59%, rgba(199, 199, 199, 1) 63%, rgba(199, 199, 199, 1) 68%, transparent 72%, transparent);
-webkit-mask-image: none;
}
input[type="radio"]::after {
content: "";
-webkit-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
background-color: rgba(245, 245, 245, 0.1);
}
input[type="radio"]::before {
content: "";
background-color: rgba(112, 186, 15, 1);
-webkit-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
width: 100%;
height: 100%;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
position: absolute;
-webkit-transform: scale(0);
-moz-transform: scale(0);
-ms-transform: scale(0);
-o-transform: scale(0);
transform: scale(0);
-webkit-transition: transform 300ms;
-moz-transition: transform 300ms;
-o-transition: transform 300ms;
-ms-transition: transform 300ms;
transition: transform 300ms;
}
input[type="radio"]:checked::before {
background-color: rgba(112, 186, 15, 1);
-webkit-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
z-index: 1;
}
input[type="radio"]:active {
background-color: transparent;
-webkit-mask-image: initial;
}
input[type="radio"]:active::after {
-webkit-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
background-color: rgba(245, 245, 245, 0.1);
}
input[type="radio"]:active:checked::before {
background-color: rgba(112, 186, 15, 1);
-webkit-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/Radio/tw_btn_radio_holo_dark.png');
}
input[type="radio"]:active:checked::after {
content: none;
}
/***************************************************************************
button
***************************************************************************/
.ui-default {
color: rgba(245, 245, 245, 1);
background-color: rgba(2, 56, 92, 1);
}
a.ui-default:active:hover,
.ui-default:enabled:active:hover {
background-color: rgba(23, 73, 115, 1);
}
.ui-default:enabled:focus {
background-color: rgba(2, 56, 92, 1);
}
a.ui-default.ui-state-disabled,
.ui-default:disabled {
color: rgba(245, 245, 245, 0.35);
background-color: rgba(3, 44, 64, 1);
}
.ui-color-red {
color: rgba(245, 245, 245, 1);
background-color: rgba(207, 36, 2, 1);
}
a.ui-color-red:active:hover,
.ui-color-red:enabled:active:hover {
background-color: rgba(222, 102, 78, 1);
}
.ui-color-red:enabled:focus {
background-color: rgba(207, 36, 2, 1);
}
a.ui-color-red.ui-state-disabled,
.ui-color-red:disabled {
color: rgba(245, 245, 245, 0.35);
background-color: rgba(69, 11, 1, 1);
}
.ui-color-orange {
color: rgba(245, 245, 245, 1);
background-color: rgba(237, 134, 0, 1);
}
a.ui-color-orange:active:hover,
.ui-color-orange:enabled:active:hover {
background-color: rgba(242, 171, 78, 1);
}
.ui-color-orange:enabled:focus {
background-color: rgba(237, 134, 0, 1);
}
a.ui-color-orange.ui-state-disabled,
.ui-color-orange:disabled {
color: rgba(245, 245, 245, 0.35);
background-color: rgba(69, 39, 5, 1);
}
.ui-color-green {
color: rgba(245, 245, 245, 1);
background-color: rgba(97, 163, 16, 1);
}
a.ui-color-green:active:hover,
.ui-color-green:enabled:active:hover {
background-color: rgba(146, 191, 90, 1);
}
.ui-color-green:enabled:focus {
background-color: rgba(97, 163, 16, 1);
}
a.ui-color-green.ui-state-disabled,
.ui-color-green:disabled {
color: rgba(245, 245, 245, 0.35);
background-color: rgba(50, 54, 48, 1);
}
.ui-grid-col-1 > .ui-btn {
border-left: solid 1px rgba(8, 8, 8, 1);
}
.ui-grid-col-2 > .ui-btn {
border-left: solid 1px rgba(8, 8, 8, 1);
}
.ui-grid-col-3 > .ui-btn {
border-left: solid 1px rgba(8, 8, 8, 1);
}
/***************************************************************************
circle progress bar
***************************************************************************/
.ui-progressbar .ui-progressbar-value .ui-progressbar-value-left {
border: 0.5rem solid rgba(18, 180, 255, 1);
}
.ui-progressbar .ui-progressbar-value .ui-progressbar-value-right {
border: 0.5rem solid rgba(18, 180, 255, 1);
}
.ui-progressbar .ui-progressbar-bg {
border: 0.5rem solid rgba(8, 81, 115, 1);
}
.ui-progressbar.ui-progressbar-small .ui-progressbar-value .ui-progressbar-value-left,
.ui-progressbar.ui-progressbar-small .ui-progressbar-value .ui-progressbar-value-right {
border: 0.25rem solid rgba(18, 180, 255, 1);
}
.ui-progressbar.ui-progressbar-small .ui-progressbar-bg {
border: 0.25rem solid rgba(8, 81, 115, 1);
}
/***************************************************************************
circular index scroll bar
***************************************************************************/
.ui-circularindexscrollbar .ui-circularindexscrollbar-indicator {
background-color: rgba(99, 99, 99, 0.9);
}
.ui-circularindexscrollbar .ui-circularindexscrollbar-indicator-text {
color: rgba(18, 180, 255, 1);
}
.ui-indexscrollbar {
display: none;
}
/***************************************************************************
drawer
***************************************************************************/
.ui-drawer {
background-color: rgba(8, 8, 8, 0.85);
border-radius: 50%;
}
.ui-drawer-overlay {
background-color: rgba(8, 8, 8, 0.85);
}
/***************************************************************************
marquee
***************************************************************************/
.ui-marquee.ui-marquee-gradient::after {
background: -webkit-linear-gradient(left, transparent 0%, rgba(18, 180, 255, 1) 100%);
}
/****************************************************
Page Indicator
*****************************************************/
.ui-page-indicator {
position: absolute;
}
.ui-page-indicator .ui-page-indicator-item {
width: 0.625rem;
height: 0.625rem;
-webkit-mask-image: -webkit-radial-gradient(#000000 0.125rem, transparent 0.25rem);
background-color: rgba(66, 66, 66, 1);
-webkit-transition-property: transform;
-moz-transition-property: transform;
-o-transition-property: transform;
-ms-transition-property: transform;
transition-property: transform;
-webkit-transition-duration: 250ms;
-moz-transition-duration: 250ms;
-o-transition-duration: 250ms;
-ms-transition-duration: 250ms;
transition-duration: 250ms;
-webkit-transition-timing-function: ease-out;
-moz-transition-timing-function: ease-out;
-o-transition-timing-function: ease-out;
-ms-transition-timing-function: ease-out;
transition-timing-function: ease-out;
}
.ui-page-indicator .ui-page-indicator-item.ui-page-indicator-active {
background-color: rgba(245, 245, 245, 1);
}
.ui-page-indicator.ui-page-indicator-linear .ui-page-indicator-item {
position: relative;
display: inline-block;
margin-right: 0.5625rem;
}
.ui-page-indicator.ui-page-indicator-linear .ui-page-indicator-item:last-child {
margin-right: 0;
}
.ui-page-indicator.ui-page-indicator-linear .ui-page-indicator-item.ui-page-indicator-active {
-webkit-transform: scale3d(1.5, 1.5, 1.5);
-moz-transform: scale3d(1.5, 1.5, 1.5);
-ms-transform: scale3d(1.5, 1.5, 1.5);
-o-transform: scale3d(1.5, 1.5, 1.5);
transform: scale3d(1.5, 1.5, 1.5);
}
/***************************************************************************
progress
***************************************************************************/
@-webkit-keyframes indeterminate {
0% {
-webkit-mask-image: url(./images/Progress/tw_progressbar_indeterminate1_holo_dark.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(18, 180, 255, 1);
}
25% {
-webkit-mask-image: url(./images/Progress/tw_progressbar_indeterminate2_holo_dark.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(18, 180, 255, 1);
}
50% {
-webkit-mask-image: url(./images/Progress/tw_progressbar_indeterminate3_holo_dark.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(18, 180, 255, 1);
}
75% {
-webkit-mask-image: url(./images/Progress/tw_progressbar_indeterminate4_holo_dark.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(18, 180, 255, 1);
}
100% {
-webkit-mask-image: url(./images/Progress/tw_progressbar_indeterminate5_holo_dark.png);
-webkit-mask-repeat: no-repeat;
background-color: rgba(18, 180, 255, 1);
}
}
progress {
color: rgba(199, 199, 199, 1);
}
progress::-webkit-progress-bar {
background-color: rgba(8, 81, 115, 1);
}
progress::-webkit-progress-value {
background-color: rgba(18, 180, 255, 1);
}
progress.ui-progress-indeterminate::-webkit-progress-bar {
background-color: rgba(18, 180, 255, 1);
}
/***********************
Processing
***********************/
.ui-processing {
background-color: rgba(18, 180, 255, 1);
}
.ui-processing.ui-processing-full-size {
background-color: rgba(18, 180, 255, 1);
}
.ui-scrollbar-bouncing-effect.ui-top {
display: none;
}
.ui-scrollbar-bouncing-effect.ui-bottom {
width: 100%;
height: 6.25rem;
left: 0;
-webkit-mask-size: 100% 6.25rem;
}
.ui-scrollbar-bouncing-effect.ui-bottom::before {
width: 100%;
height: 6.25rem;
-webkit-mask-size: 100% 6.25rem;
}
.ui-scrollbar-bouncing-effect.ui-left,
.ui-scrollbar-bouncing-effect.ui-right {
width: 6.25rem;
height: 100%;
top: 0;
-webkit-mask-size: 6.25rem 100%;
}
.ui-scrollbar-bouncing-effect.ui-left::before,
.ui-scrollbar-bouncing-effect.ui-right::before {
width: 6.25rem;
height: 100%;
-webkit-mask-size: 6.25rem 100%;
}
.ui-scrollbar-bouncing-effect.ui-bottom {
bottom: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_circle_bottom_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-bottom::before {
-webkit-mask-image: url("./images/Scroller/bouncing_circle_bottom_edge.png");
}
.ui-scrollbar-bouncing-effect.ui-left {
left: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_circle_left_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-left::before {
-webkit-mask-image: url("./images/Scroller/bouncing_circle_left_edge.png");
}
.ui-scrollbar-bouncing-effect.ui-right {
right: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_circle_right_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-right::before {
-webkit-mask-image: url("./images/Scroller/bouncing_circle_right_edge.png");
}
.is-circle-test {
width: 1px;
}
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/***************************************************************************
Body
***************************************************************************/
/***************************************************************************
Action Bar
***************************************************************************/
/****************************************************************************
Circle Progress Bar
****************************************************************************/
/****************************************************************************
Progress Bar
****************************************************************************/
/****************************************************************************
Processing
****************************************************************************/
/****************************************************************************
Toggle Switch
****************************************************************************/
/***************************************************************************
Buttons
***************************************************************************/
/***************************************************************************
Popup
***************************************************************************/
/***************************************************************************
Option Popup
***************************************************************************/
/***************************************************************************
index scrollbar (vertical)
***************************************************************************/
/***************************************************************************
circular index scrollbar
***************************************************************************/
/***************************************************************************
Listview
***************************************************************************/
/****************************************************************************
Listview with Marquee
****************************************************************************/
/***************************************************************************
Scroller
***************************************************************************/
/***************************************************************************
Swipe list
***************************************************************************/
/***************************************************************************
Checkbox/Radio
***************************************************************************/
/***************************************************************************
Page Indicator
***************************************************************************/
/***************************************************************************
More options
***************************************************************************/
/***************************************************************************
View Switcher
***************************************************************************/
/***************************************************************************
Slider
***************************************************************************/
/***************************************************************************
Selector
***************************************************************************/
/************************
Listview
*************************/
/************************************************************************/
/* fixedUIStyle.less */
/************************************************************************/
/* When system font-size is changed, winset size (ex. header, footer, button, etc,
include font) also changed. But in UX guide, some UI elements should have fixed
size regardless of system font size.
So, this less file supports fixed style for some elements.
- Layout: header, footer, button
- Popup : popup header, popup footer
- toggleswitch
- progress: text for ratio
*/
/***********************************************************************/
/* Layout(FIXED) */
/***********************************************************************/
/**********************************************************************/
/* Button(FIXED) */
/**********************************************************************/
/**********************************************************************/
/* Popup(FIXED) */
/**********************************************************************/
/**********************************************************************/
/* ToggleSwitch(FIXED) */
/**********************************************************************/
/*********************************************************************/
/* Progress(FIXED) */
/*********************************************************************/
/*
* default outline set none
*/
* {
outline: none;
}
/*
* Corrects `block` display not defined in IE 6/7/8/9 and Firefox 3.
*/
article,
aside,
details,
figcaption,
figure,
footer,
header,
hgroup,
nav,
section,
summary {
display: block;
}
/*
* Corrects `inline-block` display not defined in IE 6/7/8/9 and Firefox 3.
*/
audio,
canvas,
video {
display: inline-block;
*display: inline;
*zoom: 1;
}
/*
* Prevents modern browsers from displaying `audio` without controls.
* Remove excess height in iOS 5 devices.
*/
audio:not([controls]) {
display: none;
height: 0;
}
/*
* Addresses styling for `hidden` attribute not present in IE 7/8/9, Firefox 3,
* and Safari 4.
* Known issue: no IE 6 support.
*/
[hidden] {
display: none;
}
/* ==========================================================================
Base
========================================================================== */
/*
* 1. Corrects text resizing oddly in IE 6/7 when body `font-size` is set using
* `em` units.
* 2. Prevents iOS text size adjust after orientation change, without disabling
* user zoom.
*/
html {
font-size: 100%;
/* 1 */
-webkit-text-size-adjust: 100%;
-moz-text-size-adjust: 100%;
-ms-text-size-adjust: 100%;
-o-text-size-adjust: 100%;
text-size-adjust: 100%;
/* 2 */
}
/*
* Addresses `font-family` inconsistency between `textarea` and other form
* elements.
*/
html,
button,
input,
select,
textarea {
font-family: Tizen, Samsung Sans, Helvetica;
}
/*
* Addresses margins handled incorrectly in IE 6/7.
*/
body {
margin: 0;
}
/* ==========================================================================
Links
========================================================================== */
/*
* Improves readability when focused and also mouse hovered in all browsers.
*/
/* ==========================================================================
Typography
========================================================================== */
/*
* Addresses font sizes and margins set differently in IE 6/7.
* Addresses font sizes within `section` and `article` in Firefox 4+, Safari 5,
* and Chrome.
*/
h1 {
font-size: 2em;
margin: 0.67em 0;
}
h2 {
font-size: 1.5em;
margin: 0.83em 0;
}
h3 {
font-size: 1.17em;
margin: 1em 0;
}
h4 {
font-size: 1em;
margin: 1.33em 0;
}
h5 {
font-size: 0.83em;
margin: 1.67em 0;
}
h6 {
font-size: 0.75em;
margin: 2.33em 0;
}
/*
* Addresses styling not present in IE 7/8/9, Safari 5, and Chrome.
*/
abbr[title] {
border-bottom: 1px dotted;
}
/*
* Addresses style set to `bolder` in Firefox 3+, Safari 4/5, and Chrome.
*/
b,
strong {
font-weight: bold;
}
blockquote {
margin: 1em 40px;
}
/*
* Addresses styling not present in Safari 5 and Chrome.
*/
dfn {
font-style: italic;
}
/*
* Addresses styling not present in IE 6/7/8/9.
*/
mark {
background: #ff0;
color: #000;
}
/*
* Addresses margins set differently in IE 6/7.
*/
p,
pre {
margin: 1em 0;
}
/*
* Corrects font family set oddly in IE 6, Safari 4/5, and Chrome.
*/
code,
kbd,
pre,
samp {
font-family: monospace, serif;
_font-family: 'courier new', monospace;
font-size: 1em;
}
/*
* Improves readability of pre-formatted text in all browsers.
*/
pre {
white-space: pre;
white-space: pre-wrap;
word-wrap: break-word;
}
/*
* Addresses CSS quotes not supported in IE 6/7.
*/
q {
quotes: none;
}
/*
* Addresses `quotes` property not supported in Safari 4.
*/
q:before,
q:after {
content: '';
content: none;
}
small {
font-size: 75%;
}
/*
* Prevents `sub` and `sup` affecting `line-height` in all browsers.
*/
sub,
sup {
font-size: 75%;
line-height: 0;
position: relative;
vertical-align: baseline;
}
sup {
top: -0.5em;
}
sub {
bottom: -0.25em;
}
/* ==========================================================================
Lists
========================================================================== */
/*
* Addresses margins set differently in IE 6/7.
*/
dl,
menu,
ol,
ul {
margin: 1em 0;
}
dd {
margin: 0 0 0 40px;
}
/*
* Addresses paddings set differently in IE 6/7.
*/
menu,
ol,
ul {
padding: 0 0 0 40px;
}
/*
* Corrects list images handled incorrectly in IE 7.
*/
nav ul,
nav ol {
list-style: none;
list-style-image: none;
}
/* ==========================================================================
Embedded content
========================================================================== */
/*
* 1. Removes border when inside `a` element in IE 6/7/8/9 and Firefox 3.
* 2. Improves image quality when scaled in IE 7.
*/
img {
border: 0;
/* 1 */
-ms-interpolation-mode: bicubic;
/* 2 */
}
/*
* Corrects overflow displayed oddly in IE 9.
*/
svg:not(:root) {
overflow: hidden;
}
/* ==========================================================================
Figures
========================================================================== */
/*
* Addresses margin not present in IE 6/7/8/9, Safari 5, and Opera 11.
*/
figure {
margin: 0;
}
/* ==========================================================================
Forms
========================================================================== */
/*
* Corrects margin displayed oddly in IE 6/7.
*/
form {
margin: 0;
}
/*
* Define consistent border, margin, and padding.
*/
fieldset {
border: 1px solid #c0c0c0;
margin: 0 2px;
padding: 0.35em 0.625em 0.75em;
}
/*
* 1. Corrects color not being inherited in IE 6/7/8/9.
* 2. Corrects text not wrapping in Firefox 3.
* 3. Corrects alignment displayed oddly in IE 6/7.
*/
legend {
border: 0;
/* 1 */
padding: 0;
white-space: normal;
/* 2 */
*margin-left: -7px;
/* 3 */
}
/*
* 1. Corrects font size not being inherited in all browsers.
* 2. Addresses margins set differently in IE 6/7, Firefox 3+, Safari 5,
* and Chrome.
* 3. Improves appearance and consistency in all browsers.
*/
button,
input,
select,
textarea {
font-size: 100%;
/* 1 */
margin: 0;
/* 2 */
vertical-align: baseline;
/* 3 */
*vertical-align: middle;
/* 3 */
}
/*
* Addresses Firefox 3+ setting `line-height` on `input` using `!important` in
* the UA stylesheet.
*/
button,
input {
line-height: normal;
}
/*
* Re-set default cursor for disabled elements.
*/
button[disabled],
input[disabled] {
cursor: default;
}
input[type="time"],
input[type="date"] {
color: #000000;
}
/*
* 1. Addresses `appearance` set to `searchfield` in Safari 5 and Chrome.
* 2. Addresses `box-sizing` set to `border-box` in Safari 5 and Chrome
* (include `-moz` to future-proof).
*/
input[type="search"] {
-webkit-appearance: textfield;
-moz-appearance: textfield;
-ms-appearance: textfield;
-o-appearance: textfield;
appearance: textfield;
-webkit-box-sizing: content-box;
-moz-box-sizing: content-box;
box-sizing: content-box;
}
/*
* Removes inner padding and search cancel button in Safari 5 and Chrome
* on OS X.
*/
input[type="search"]::-webkit-search-cancel-button,
input[type="search"]::-webkit-search-decoration {
-webkit-appearance: none;
-moz-appearance: none;
-ms-appearance: none;
-o-appearance: none;
appearance: none;
}
/*
* Removes inner padding and border in Firefox 3+.
*/
button::-moz-focus-inner,
input::-moz-focus-inner {
border: 0;
padding: 0;
}
/*
* 1. Removes default vertical scrollbar in IE 6/7/8/9.
* 2. Improves readability and alignment in all browsers.
*/
textarea {
overflow: auto;
/* 1 */
vertical-align: top;
/* 2 */
}
/* ==========================================================================
Tables
========================================================================== */
/*
* Remove most spacing between table cells.
*/
table {
border-collapse: collapse;
border-spacing: 0;
}
/*
* Common definition for theme
*/
/***************************************************************
default font size (base font from WRT)
=> small: 13px
=> normal: 17px
=> large: 20px
This is only applied to gear2.
so, we set @font_size_default as 17px on gear2 and it was changed to 16px since gear-S.
This value only used for @rem_base, not for html font-size
html font-size is set by WRT base font-size
***************************************************************/
/******************************
z-index order collection
******************************/
/***************************************************************************
Common Style
***************************************************************************/
.tau-info-theme {
position: absolute;
top: -999px;
left: -999px;
}
.screen-hidden {
visibility: hidden;
position: absolute;
top: -10000em;
left: -10000em;
}
/***************************************************************************
Default Style.
***************************************************************************/
button,
input {
outline: none;
cursor: pointer;
}
* {
-webkit-box-sizing: border-box;
-moz-box-sizing: border-box;
box-sizing: border-box;
}
a {
text-decoration: none !important;
color: rgba(245, 245, 245, 1);
}
ul,
li {
margin: 0;
padding: 0;
list-style-type: none;
}
img {
margin: 0;
padding: 0;
}
/******************************
Global LESS mixin collection
******************************/
/***************************************************************************
Common Style
***************************************************************************/
html {
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
}
body {
height: 100%;
font-size: 2.125rem;
font-family: Tizen, Samsung Sans, Helvetica;
background: rgba(8, 8, 8, 1);
color: rgba(245, 245, 245, 1);
}
.ui-page {
width: 100%;
height: 100%;
overflow: hidden;
position: absolute;
top: 0;
left: 0;
display: none;
background: rgba(8, 8, 8, 1);
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
}
.ui-page.ui-page-active {
display: flex;
display: -webkit-flex;
flex-direction: column;
-webkit-flex-direction: column;
}
.ui-page.ui-page-build {
display: flex;
display: -webkit-flex;
flex-direction: column;
-webkit-flex-direction: column;
visibility: hidden;
}
.ui-content {
width: 100%;
height: auto;
overflow: auto;
-webkit-overflow-scrolling: touch;
-moz-overflow-scrolling: touch;
-o-overflow-scrolling: touch;
-ms-overflow-scrolling: touch;
overflow-scrolling: touch;
flex: 1;
-webkit-flex: 1;
}
.ui-header {
position: relative;
width: 100%;
height: 60px;
line-height: 60px;
background-color: rgba(0, 31, 56, 1);
}
.ui-header .ui-title {
color: rgba(116, 196, 252, 1);
font-size: 32px;
font-weight: bold;
white-space: nowrap;
overflow: hidden;
margin: 0;
padding-left: 16px;
padding-right: 16px;
-webkit-mask-image: -webkit-linear-gradient(left, #000000 80%, rgba(0, 0, 0, 0) 95%);
}
.ui-header .ui-title.ui-icon {
padding-left: 60px;
background-position: 16px 14px;
background-size: 34px 34px;
background-repeat: no-repeat;
}
.ui-header.ui-has-more .ui-title {
padding-right: 78px;
-webkit-mask-image: -webkit-linear-gradient(left, #000000 70%, rgba(0, 0, 0, 0) 75%);
}
.ui-header button.ui-more,
.ui-header input.ui-more {
border: 0 none;
padding: 0;
}
.ui-header .ui-more-disable {
cursor: default;
background-color: transparent;
}
.ui-header .ui-more {
display: block;
position: absolute;
top: 0;
right: 0;
width: 68px;
height: 100%;
overflow: hidden;
text-indent: -1000em;
background-position: center center;
background-color: transparent;
cursor: pointer;
}
.ui-header .ui-more:active {
background-color: rgba(23, 73, 115, 1);
}
.ui-header .ui-more.ui-icon-detail {
background-color: rgba(116, 196, 252, 1);
-webkit-mask-image: url("./images/Actionbar/tw_ic_menu_detail_normal_holo_dark.png");
-moz-mask-image: url("./images/Actionbar/tw_ic_menu_detail_normal_holo_dark.png");
-ms-mask-image: url("./images/Actionbar/tw_ic_menu_detail_normal_holo_dark.png");
-o-mask-image: url("./images/Actionbar/tw_ic_menu_detail_normal_holo_dark.png");
mask-image: url("./images/Actionbar/tw_ic_menu_detail_normal_holo_dark.png");
-webkit-mask-size: 3.25rem 3.75rem;
-moz-mask-size: 3.25rem 3.75rem;
-ms-mask-size: 3.25rem 3.75rem;
-o-mask-size: 3.25rem 3.75rem;
mask-size: 3.25rem 3.75rem;
-webkit-mask-position: center;
-moz-mask-position: center;
-ms-mask-position: center;
-o-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-header .ui-more.ui-icon-detail[disabled="disabled"] {
cursor: default;
background-color: transparent;
background-color: rgba(116, 196, 252, 0.2);
-webkit-mask-image: url("./images/Actionbar/tw_ic_menu_detail_disable_holo_dark.png");
-moz-mask-image: url("./images/Actionbar/tw_ic_menu_detail_disable_holo_dark.png");
-ms-mask-image: url("./images/Actionbar/tw_ic_menu_detail_disable_holo_dark.png");
-o-mask-image: url("./images/Actionbar/tw_ic_menu_detail_disable_holo_dark.png");
mask-image: url("./images/Actionbar/tw_ic_menu_detail_disable_holo_dark.png");
-webkit-mask-size: 3.25rem 3.75rem;
-moz-mask-size: 3.25rem 3.75rem;
-ms-mask-size: 3.25rem 3.75rem;
-o-mask-size: 3.25rem 3.75rem;
mask-size: 3.25rem 3.75rem;
-webkit-mask-position: center;
-moz-mask-position: center;
-ms-mask-position: center;
-o-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-header .ui-more.ui-icon-overflow {
background-color: rgba(116, 196, 252, 1);
-webkit-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-moz-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-ms-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-o-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-webkit-mask-size: 3.25rem 3.75rem;
-moz-mask-size: 3.25rem 3.75rem;
-ms-mask-size: 3.25rem 3.75rem;
-o-mask-size: 3.25rem 3.75rem;
mask-size: 3.25rem 3.75rem;
-webkit-mask-position: center;
-moz-mask-position: center;
-ms-mask-position: center;
-o-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-header .ui-more.ui-icon-overflow[disabled="disabled"] {
cursor: default;
background-color: transparent;
background-color: rgba(116, 196, 252, 0.2);
-webkit-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-moz-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-ms-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-o-mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
mask-image: url("./images/Actionbar/tw_ic_menu_moreoverflow_holo_dark.png");
-webkit-mask-size: 3.25rem 3.75rem;
-moz-mask-size: 3.25rem 3.75rem;
-ms-mask-size: 3.25rem 3.75rem;
-o-mask-size: 3.25rem 3.75rem;
mask-size: 3.25rem 3.75rem;
-webkit-mask-position: center;
-moz-mask-position: center;
-ms-mask-position: center;
-o-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-header .ui-more.ui-icon-selectall {
background-color: rgba(116, 196, 252, 1);
-webkit-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_normal_holo_dark.png");
-moz-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_normal_holo_dark.png");
-ms-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_normal_holo_dark.png");
-o-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_normal_holo_dark.png");
mask-image: url("./images/Actionbar/tw_ic_menu_selectall_normal_holo_dark.png");
-webkit-mask-size: 3.25rem 3.75rem;
-moz-mask-size: 3.25rem 3.75rem;
-ms-mask-size: 3.25rem 3.75rem;
-o-mask-size: 3.25rem 3.75rem;
mask-size: 3.25rem 3.75rem;
-webkit-mask-position: center;
-moz-mask-position: center;
-ms-mask-position: center;
-o-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-header .ui-more.ui-icon-selectall[disabled="disabled"] {
cursor: default;
background-color: transparent;
background-color: rgba(116, 196, 252, 0.2);
-webkit-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_disable_holo_dark.png");
-moz-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_disable_holo_dark.png");
-ms-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_disable_holo_dark.png");
-o-mask-image: url("./images/Actionbar/tw_ic_menu_selectall_disable_holo_dark.png");
mask-image: url("./images/Actionbar/tw_ic_menu_selectall_disable_holo_dark.png");
-webkit-mask-size: 3.25rem 3.75rem;
-moz-mask-size: 3.25rem 3.75rem;
-ms-mask-size: 3.25rem 3.75rem;
-o-mask-size: 3.25rem 3.75rem;
mask-size: 3.25rem 3.75rem;
-webkit-mask-position: center;
-moz-mask-position: center;
-ms-mask-position: center;
-o-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-footer {
width: 100%;
height: 85px;
position: relative;
bottom: 0;
left: 0;
}
.ui-footer .ui-btn.ui-btn-icon-only {
position: relative;
text-indent: -624.9375rem;
}
.ui-footer .ui-btn.ui-btn-icon-only::before {
content: "";
position: absolute;
-webkit-mask-size: 3.125rem 3.125rem;
-webkit-mask-repeat: no-repeat;
width: 3.125rem;
height: 3.125rem;
top: 50%;
left: 50%;
transform: translate3d(-50%, -50%, 0);
}
/***************************************************************************
Page Scroll
***************************************************************************/
.ui-page.ui-scroll-on .ui-scroller {
width: 100%;
height: auto;
overflow: hidden;
overflow-y: auto;
-webkit-overflow-scrolling: touch;
-moz-overflow-scrolling: touch;
-o-overflow-scrolling: touch;
-ms-overflow-scrolling: touch;
overflow-scrolling: touch;
flex: 1;
-webkit-flex: 1;
}
.ui-page.ui-scroll-on .ui-content {
height: auto !important;
flex: none;
-webkit-flex: none;
}
.ui-page.ui-scroll-on .ui-footer {
position: relative;
}
.ui-footer.ui-fixed {
position: fixed !important;
bottom: 0;
left: 0;
width: 100%;
-webkit-transform: translate3d(0, 0, 0);
}
/***************************************************************************
Grid layout
***************************************************************************/
.ui-grid-col > * {
float: left;
}
.ui-page .ui-grid-col-1 > * {
float: left;
}
.ui-page .ui-grid-col-1 > * {
width: 100%;
}
.ui-page .ui-grid-col-2 > * {
float: left;
}
.ui-page .ui-grid-col-2 > * {
width: 50%;
}
.ui-page .ui-grid-col-3 > * {
float: left;
}
.ui-page .ui-grid-col-3 > * {
width: 33.333333333333336%;
}
.ui-grid-row > * {
display: block;
}
/*
* Progressbar
*/
@-webkit-keyframes indeterminate {
0% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
25% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
50% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
75% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
100% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
}
@-moz-keyframes indeterminate {
0% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
25% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
50% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
75% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
100% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
}
@-ms-keyframes indeterminate {
0% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
25% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
50% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
75% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
100% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
}
@-o-keyframes indeterminate {
0% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
25% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
50% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
75% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
100% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
}
@keyframes indeterminate {
0% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate1_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
25% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate2_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
50% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate3_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
75% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate4_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
100% {
-webkit-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-moz-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-ms-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-o-mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
mask-image: url('./images/Progress/tw_progressbar_indeterminate5_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: rgba(0, 77, 153, 1);
}
}
progress {
position: relative;
padding-left: 1rem;
padding-right: 1rem;
margin-top: 0.875rem;
margin-bottom: 0.875rem;
width: 100%;
height: 0.375rem;
-webkit-appearance: none;
-moz-appearance: none;
-ms-appearance: none;
-o-appearance: none;
appearance: none;
border: none;
}
progress::-webkit-progress-bar {
position: relative;
background-color: rgba(71, 71, 71, 1);
border-radius: 0.125rem;
overflow: hidden;
}
progress::-webkit-progress-value {
background-color: rgba(55, 161, 237, 1);
height: 0.375rem;
border-radius: 0.125rem;
overflow: hidden;
}
progress.ui-progress-indeterminate::-webkit-progress-bar {
background-color: rgba(55, 161, 237, 1);
}
progress.ui-progress-indeterminate::-webkit-progress-value {
background-size: 100% 100%;
-webkit-animation: indeterminate 150ms infinite;
-moz-animation: indeterminate 150ms infinite;
-ms-animation: indeterminate 150ms infinite;
-o-animation: indeterminate 150ms infinite;
animation: indeterminate 150ms infinite;
}
.ui-progress-proportion {
display: inline-block;
float: left;
font-size: 20px;
padding-left: 1rem;
padding-right: 1rem;
margin-top: -0.625rem;
color: rgba(245, 245, 245, 1);
}
.ui-progress-ratio {
display: inline-block;
float: right;
font-size: 20px;
padding-left: 1rem;
padding-right: 1rem;
margin-top: -0.625rem;
color: rgba(245, 245, 245, 1);
}
.ui-circle-progress {
display: none;
}
.ui-progressbar.ui-progressbar-full {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
}
.ui-progressbar.ui-progressbar-large {
width: 6.25rem;
height: 6.25rem;
}
.ui-progressbar.ui-progressbar-medium {
width: 5.25rem;
height: 5.25rem;
}
.ui-progressbar.ui-progressbar-small {
width: 3.5rem;
height: 3.5rem;
}
.ui-progressbar {
position: relative;
display: inline-block;
pointer-events: none;
}
.ui-progressbar .ui-progressbar-value.ui-progressbar-half {
clip: rect(auto, auto, auto, auto);
}
.ui-progressbar .ui-progressbar-value.ui-progressbar-half .ui-progressbar-value-right {
-webkit-transform: rotate(180deg);
}
.ui-progressbar .ui-progressbar-value {
clip: rect(0, 1em, 1em, 0.5em);
width: 100%;
height: 100%;
position: absolute;
}
.ui-progressbar .ui-progressbar-value .ui-progressbar-value-left {
border: 0.5rem solid rgba(55, 161, 237, 1);
border-radius: 50%;
clip: rect(0, 0.5em, 1em, 0);
height: 100%;
width: 100%;
position: absolute;
}
.ui-progressbar .ui-progressbar-value .ui-progressbar-value-right {
border: 0.5rem solid rgba(55, 161, 237, 1);
border-radius: 50%;
clip: rect(0, 0.5em, 1em, 0);
width: 100%;
height: 100%;
position: absolute;
}
.ui-progressbar .ui-progressbar-bg {
border: 0.5rem solid rgba(71, 71, 71, 1);
border-radius: 50%;
width: 100%;
height: 100%;
}
/*
* toggle switch
*/
.ui-switch {
width: 100%;
color: rgba(245, 245, 245, 1);
}
.ui-switch-inneroffset,
.ui-switch-handler {
display: none;
}
.ui-switch-text {
margin-top: 30px;
margin-bottom: 40px;
padding: 0 30px;
font-size: 34px;
text-align: center;
}
.ui-switch-sub-text {
margin-top: 48px;
font-size: 26px;
text-align: center;
color: T161;
}
.ui-toggleswitch {
position: relative;
display: inline-block;
width: 103.5px;
height: 55px;
overflow: hidden;
}
.ui-toggleswitch.ui-toggleswitch-large {
width: 207px;
height: 110px;
}
.ui-toggleswitch .ui-switch-input {
position: absolute;
width: 100%;
height: 100%;
-webkit-appearance: none;
display: block;
}
.ui-toggleswitch .ui-switch-input::before {
content: "";
position: absolute;
width: 100%;
height: 100%;
-webkit-mask-image: url('images/Controller_icon/toggle_changeable.png');
-moz-mask-image: url('images/Controller_icon/toggle_changeable.png');
-ms-mask-image: url('images/Controller_icon/toggle_changeable.png');
-o-mask-image: url('images/Controller_icon/toggle_changeable.png');
mask-image: url('images/Controller_icon/toggle_changeable.png');
-webkit-mask-size: 100% 400%;
-moz-mask-size: 100% 400%;
-ms-mask-size: 100% 400%;
-o-mask-size: 100% 400%;
mask-size: 100% 400%;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-position: 0 0;
-moz-mask-position: 0 0;
-ms-mask-position: 0 0;
-o-mask-position: 0 0;
mask-position: 0 0;
background-color: rgba(97, 96, 96, 1);
z-index: 0;
}
.ui-toggleswitch .ui-switch-input:active::before {
background-color: rgba(97, 96, 96, 1);
}
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-activation,
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-button {
background-color: rgba(173, 173, 173, 1);
}
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-input:active ~ .ui-switch-button::before {
background-color: rgba(97, 96, 96, 1);
}
.ui-toggleswitch .ui-switch-activation,
.ui-toggleswitch .ui-switch-button {
width: 100%;
height: 100%;
-webkit-mask-image: url('images/Controller_icon/toggle_changeable.png');
-moz-mask-image: url('images/Controller_icon/toggle_changeable.png');
-ms-mask-image: url('images/Controller_icon/toggle_changeable.png');
-o-mask-image: url('images/Controller_icon/toggle_changeable.png');
mask-image: url('images/Controller_icon/toggle_changeable.png');
-webkit-mask-size: 100% 400%;
-moz-mask-size: 100% 400%;
-ms-mask-size: 100% 400%;
-o-mask-size: 100% 400%;
mask-size: 100% 400%;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-position: 0 33.3%;
-moz-mask-position: 0 33.3%;
-ms-mask-position: 0 33.3%;
-o-mask-position: 0 33.3%;
mask-position: 0 33.3%;
background-color: rgba(245, 245, 245, 1);
position: absolute;
top: 0;
left: 0;
pointer-events: none;
z-index: 1;
}
.ui-toggleswitch .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-button::before {
content: "";
position: absolute;
width: 100%;
height: 100%;
-webkit-mask-image: url('images/Controller_icon/toggle_changeable.png');
-moz-mask-image: url('images/Controller_icon/toggle_changeable.png');
-ms-mask-image: url('images/Controller_icon/toggle_changeable.png');
-o-mask-image: url('images/Controller_icon/toggle_changeable.png');
mask-image: url('images/Controller_icon/toggle_changeable.png');
-webkit-mask-size: 100% 400%;
-moz-mask-size: 100% 400%;
-ms-mask-size: 100% 400%;
-o-mask-size: 100% 400%;
mask-size: 100% 400%;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-position: 0 66.7%;
-moz-mask-position: 0 66.7%;
-ms-mask-position: 0 66.7%;
-o-mask-position: 0 66.7%;
mask-position: 0 66.7%;
background-color: rgba(128, 128, 128, 1);
top: 0;
left: 0;
}
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-activation,
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-button {
width: 100%;
height: 100%;
background-color: rgba(245, 245, 245, 1);
-webkit-mask-position: 0 33%;
-moz-mask-position: 0 33%;
-ms-mask-position: 0 33%;
-o-mask-position: 0 33%;
mask-position: 0 33%;
-webkit-transform: translate3d(36.23%, 0, 0);
-moz-transform: translate3d(36.23%, 0, 0);
-ms-transform: translate3d(36.23%, 0, 0);
-o-transform: translate3d(36.23%, 0, 0);
transform: translate3d(36.23%, 0, 0);
}
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-input:checked ~ .ui-switch-button::before {
-webkit-mask-position: 0 100%;
-moz-mask-position: 0 100%;
-ms-mask-position: 0 100%;
-o-mask-position: 0 100%;
mask-position: 0 100%;
background-color: rgba(112, 186, 15, 1);
}
.ui-toggleswitch .ui-switch-input:checked::before {
background-color: rgba(112, 186, 15, 1);
}
.ui-toggleswitch .ui-switch-input:checked:active::before {
background-color: rgba(71, 97, 37, 1);
}
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-activation,
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-button {
background-color: rgba(173, 173, 173, 1);
}
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-activation::before,
.ui-toggleswitch .ui-switch-input:checked:active ~ .ui-switch-button::before {
background-color: rgba(71, 97, 37, 1);
}
/***********************
Processing
***********************/
@-webkit-keyframes fullprocess {
0% {
-webkit-mask-position: 0vw 0vw;
}
2% {
-webkit-mask-position: -100vw 0vw;
}
3% {
-webkit-mask-position: -200vw 0vw;
}
5% {
-webkit-mask-position: -300vw 0vw;
}
7% {
-webkit-mask-position: -400vw 0vw;
}
8% {
-webkit-mask-position: -500vw 0vw;
}
10% {
-webkit-mask-position: -600vw 0vw;
}
12% {
-webkit-mask-position: 0vw -100vw;
}
14% {
-webkit-mask-position: -100vw -100vw;
}
15% {
-webkit-mask-position: -200vw -100vw;
}
17% {
-webkit-mask-position: -300vw -100vw;
}
19% {
-webkit-mask-position: -400vw -100vw;
}
20% {
-webkit-mask-position: -500vw -100vw;
}
22% {
-webkit-mask-position: -600vw -100vw;
}
24% {
-webkit-mask-position: 0vw -200vw;
}
25% {
-webkit-mask-position: -100vw -200vw;
}
27% {
-webkit-mask-position: -200vw -200vw;
}
29% {
-webkit-mask-position: -300vw -200vw;
}
31% {
-webkit-mask-position: -400vw -200vw;
}
32% {
-webkit-mask-position: -500vw -200vw;
}
34% {
-webkit-mask-position: -600vw -200vw;
}
36% {
-webkit-mask-position: 0vw -300vw;
}
37% {
-webkit-mask-position: -100vw -300vw;
}
39% {
-webkit-mask-position: -200vw -300vw;
}
41% {
-webkit-mask-position: -300vw -300vw;
}
42% {
-webkit-mask-position: -400vw -300vw;
}
44% {
-webkit-mask-position: -500vw -300vw;
}
46% {
-webkit-mask-position: -600vw -300vw;
}
47% {
-webkit-mask-position: 0vw -400vw;
}
49% {
-webkit-mask-position: -100vw -400vw;
}
51% {
-webkit-mask-position: -200vw -400vw;
}
53% {
-webkit-mask-position: -300vw -400vw;
}
54% {
-webkit-mask-position: -400vw -400vw;
}
56% {
-webkit-mask-position: -500vw -400vw;
}
58% {
-webkit-mask-position: -600vw -400vw;
}
59% {
-webkit-mask-position: 0vw -500vw;
}
61% {
-webkit-mask-position: -100vw -500vw;
}
63% {
-webkit-mask-position: -200vw -500vw;
}
64% {
-webkit-mask-position: -300vw -500vw;
}
66% {
-webkit-mask-position: -400vw -500vw;
}
68% {
-webkit-mask-position: -500vw -500vw;
}
69% {
-webkit-mask-position: -600vw -500vw;
}
71% {
-webkit-mask-position: 0vw -600vw;
}
73% {
-webkit-mask-position: -100vw -600vw;
}
75% {
-webkit-mask-position: -200vw -600vw;
}
76% {
-webkit-mask-position: -300vw -600vw;
}
78% {
-webkit-mask-position: -400vw -600vw;
}
80% {
-webkit-mask-position: -500vw -600vw;
}
81% {
-webkit-mask-position: -600vw -600vw;
}
83% {
-webkit-mask-position: 0vw -700vw;
}
85% {
-webkit-mask-position: -100vw -700vw;
}
86% {
-webkit-mask-position: -200vw -700vw;
}
88% {
-webkit-mask-position: -300vw -700vw;
}
90% {
-webkit-mask-position: -400vw -700vw;
}
92% {
-webkit-mask-position: -500vw -700vw;
}
93% {
-webkit-mask-position: -600vw -700vw;
}
95% {
-webkit-mask-position: 0vw -800vw;
}
97% {
-webkit-mask-position: -100vw -800vw;
}
98% {
-webkit-mask-position: -200vw -800vw;
}
100% {
-webkit-mask-position: -300vw -800vw;
}
}
@-webkit-keyframes smallprocess {
0% {
-webkit-mask-position: 0px 0px;
}
2% {
-webkit-mask-position: -76px 0px;
}
3% {
-webkit-mask-position: -152px 0px;
}
5% {
-webkit-mask-position: -228px 0px;
}
7% {
-webkit-mask-position: -304px 0px;
}
8% {
-webkit-mask-position: -380px 0px;
}
10% {
-webkit-mask-position: -456px 0px;
}
12% {
-webkit-mask-position: 0px -76px;
}
14% {
-webkit-mask-position: -76px -76px;
}
15% {
-webkit-mask-position: -152px -76px;
}
17% {
-webkit-mask-position: -228px -76px;
}
19% {
-webkit-mask-position: -304px -76px;
}
20% {
-webkit-mask-position: -380px -76px;
}
22% {
-webkit-mask-position: -456px -76px;
}
24% {
-webkit-mask-position: 0px -152px;
}
25% {
-webkit-mask-position: -76px -152px;
}
27% {
-webkit-mask-position: -152px -152px;
}
29% {
-webkit-mask-position: -228px -152px;
}
31% {
-webkit-mask-position: -304px -152px;
}
32% {
-webkit-mask-position: -380px -152px;
}
34% {
-webkit-mask-position: -456px -152px;
}
36% {
-webkit-mask-position: 0px -228px;
}
37% {
-webkit-mask-position: -76px -228px;
}
39% {
-webkit-mask-position: -152px -228px;
}
41% {
-webkit-mask-position: -228px -228px;
}
42% {
-webkit-mask-position: -304px -228px;
}
44% {
-webkit-mask-position: -380px -228px;
}
46% {
-webkit-mask-position: -456px -228px;
}
47% {
-webkit-mask-position: 0px -304px;
}
49% {
-webkit-mask-position: -76px -304px;
}
51% {
-webkit-mask-position: -152px -304px;
}
53% {
-webkit-mask-position: -228px -304px;
}
54% {
-webkit-mask-position: -304px -304px;
}
56% {
-webkit-mask-position: -380px -304px;
}
58% {
-webkit-mask-position: -456px -304px;
}
59% {
-webkit-mask-position: 0px -380px;
}
61% {
-webkit-mask-position: -76px -380px;
}
63% {
-webkit-mask-position: -152px -380px;
}
64% {
-webkit-mask-position: -228px -380px;
}
66% {
-webkit-mask-position: -304px -380px;
}
68% {
-webkit-mask-position: -380px -380px;
}
69% {
-webkit-mask-position: -456px -380px;
}
71% {
-webkit-mask-position: 0px -456px;
}
73% {
-webkit-mask-position: -76px -456px;
}
75% {
-webkit-mask-position: -152px -456px;
}
76% {
-webkit-mask-position: -228px -456px;
}
78% {
-webkit-mask-position: -304px -456px;
}
80% {
-webkit-mask-position: -380px -456px;
}
81% {
-webkit-mask-position: -456px -456px;
}
83% {
-webkit-mask-position: 0px -532px;
}
85% {
-webkit-mask-position: -76px -532px;
}
86% {
-webkit-mask-position: -152px -532px;
}
88% {
-webkit-mask-position: -228px -532px;
}
90% {
-webkit-mask-position: -304px -532px;
}
92% {
-webkit-mask-position: -380px -532px;
}
93% {
-webkit-mask-position: -456px -532px;
}
95% {
-webkit-mask-position: 0px -608px;
}
97% {
-webkit-mask-position: -76px -608px;
}
98% {
-webkit-mask-position: -152px -608px;
}
100% {
-webkit-mask-position: -228px -608px;
}
}
.ui-processing {
background-color: rgba(0, 77, 153, 1);
-webkit-mask-image: url("./images/Processing/processingsmall.png");
-webkit-mask-repeat: no-repeat;
-webkit-animation-duration: 1s;
-webkit-animation-iteration-count: infinite;
-webkit-animation-timing-function: step-start;
-webkit-animation-name: smallprocess;
width: 4.75rem;
height: 4.75rem;
margin: 0 auto;
}
.ui-processing-text {
margin: 0 auto;
text-align: center;
width: 10.5rem;
font-size: 1.5rem;
color: rgba(245, 245, 245, 1);
}
.ui-processing.ui-processing-full-size {
background-color: rgba(0, 77, 153, 1);
-webkit-mask-image: url("./images/Processing/processingfull.png");
-webkit-mask-repeat: no-repeat;
-webkit-animation-duration: 1s;
-webkit-animation-iteration-count: infinite;
-webkit-animation-timing-function: step-start;
-webkit-animation-name: fullprocess;
pointer-events: none;
position: fixed;
width: 100vw;
height: 100vw;
top: 0;
left: 0;
margin: 0;
}
.ui-btn {
display: block;
padding-top: 28.5px;
padding-right: 10px;
padding-bottom: 28.5px;
padding-left: 10px;
margin: 0px;
font-size: 32px;
line-height: 28px;
text-align: center;
white-space: nowrap;
cursor: pointer;
vertical-align: middle;
text-overflow: ellipsis;
overflow: hidden;
border: 0 none;
color: rgba(245, 245, 245, 1);
background-color: rgba(2, 56, 92, 1);
}
a.ui-btn:active:hover,
.ui-btn:enabled:active:hover {
background-color: rgba(23, 73, 115, 1);
}
.ui-btn:enabled:focus {
background-color: rgba(2, 56, 92, 1);
}
a.ui-btn.ui-state-disabled,
.ui-btn:disabled {
color: rgba(245, 245, 245, 0.2);
background-color: rgba(3, 44, 64, 1);
cursor: default;
}
.ui-btn.ui-multiline {
padding-top: 10px;
padding-bottom: 10px;
}
.ui-btn.ui-inline {
display: inline-block;
}
.ui-btn.ui-btn-icon {
background-position: center center;
height: 76px;
text-indent: -1000em;
overflow: hidden;
background-repeat: no-repeat;
background-size: 60px 60px;
position: relative;
}
.ui-btn-icon::before {
content: "";
background-color: rgba(245, 245, 245, 1);
-webkit-mask-size: 60px 60px;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: 0 0;
width: 60px;
height: 60px;
top: 50%;
left: 50%;
transform: translate3d(-50%, -50%, 0);
position: absolute;
-webkit-mask-image: -webkit-linear-gradient(transparent, transparent);
}
.ui-default {
color: rgba(245, 245, 245, 1);
background-color: rgba(2, 56, 92, 1);
}
a.ui-default:active:hover,
.ui-default:enabled:active:hover {
background-color: rgba(23, 73, 115, 1);
}
.ui-default:enabled:focus {
background-color: rgba(2, 56, 92, 1);
}
a.ui-default.ui-state-disabled,
.ui-default:disabled {
color: rgba(245, 245, 245, 0.2);
background-color: rgba(3, 44, 64, 1);
cursor: default;
}
.ui-color-red {
color: rgba(245, 245, 245, 1);
background-color: rgba(207, 36, 2, 1);
}
a.ui-color-red:active:hover,
.ui-color-red:enabled:active:hover {
background-color: rgba(222, 102, 78, 1);
}
.ui-color-red:enabled:focus {
background-color: rgba(207, 36, 2, 1);
}
a.ui-color-red.ui-state-disabled,
.ui-color-red:disabled {
color: rgba(245, 245, 245, 0.2);
background-color: rgba(69, 11, 1, 1);
cursor: default;
}
.ui-color-orange {
color: rgba(245, 245, 245, 1);
background-color: rgba(237, 134, 0, 1);
}
a.ui-color-orange:active:hover,
.ui-color-orange:enabled:active:hover {
background-color: rgba(242, 171, 78, 1);
}
.ui-color-orange:enabled:focus {
background-color: rgba(237, 134, 0, 1);
}
a.ui-color-orange.ui-state-disabled,
.ui-color-orange:disabled {
color: rgba(245, 245, 245, 0.2);
background-color: rgba(69, 39, 5, 1);
cursor: default;
}
.ui-color-green {
color: rgba(245, 245, 245, 1);
background-color: rgba(97, 163, 16, 1);
}
a.ui-color-green:active:hover,
.ui-color-green:enabled:active:hover {
background-color: rgba(146, 191, 90, 1);
}
.ui-color-green:enabled:focus {
background-color: rgba(97, 163, 16, 1);
}
a.ui-color-green.ui-state-disabled,
.ui-color-green:disabled {
color: rgba(245, 245, 245, 0.2);
background-color: rgba(50, 54, 48, 1);
cursor: default;
}
.ui-btn:focus,
.ui-btn:active {
outline: none;
}
a.ui-btn {
text-decoration: none;
box-sizing: border-box;
}
button.ui-btn,
input.ui-btn {
box-sizing: border-box;
-webkit-appearance: none;
-moz-appearance: none;
-ms-appearance: none;
-o-appearance: none;
appearance: none;
width: 100%;
}
button.ui-btn.ui-inline,
input.ui-btn.ui-inline {
display: inline-block;
width: auto;
}
.ui-grid-col-1 > .ui-btn {
border-left: solid 1px rgba(8, 8, 8, 1);
}
.ui-grid-col-1 > .ui-btn:first-child {
border-left: 0px none;
}
.ui-grid-col-2 > .ui-btn {
border-left: solid 1px rgba(8, 8, 8, 1);
}
.ui-grid-col-2 > .ui-btn:first-child {
border-left: 0px none;
}
.ui-grid-col-3 > .ui-btn {
border-left: solid 1px rgba(8, 8, 8, 1);
}
.ui-grid-col-3 > .ui-btn:first-child {
border-left: 0px none;
}
.ui-grid-row > .ui-btn {
margin-top: 1px;
}
.ui-grid-row > .ui-btn:first-child {
margin-top: 0px;
}
/***********************
Normal Popup
***********************/
.ui-popup {
position: fixed;
left: 0px;
width: 100%;
margin: 0;
background-color: rgba(0, 17, 33, 1);
border: 3px solid rgba(0, 55, 92, 1);
display: none;
z-index: 1100;
}
.ui-popup:not(.ui-ctxpopup) {
bottom: 0;
}
.ui-popup .ui-popup-header {
width: 100%;
min-height: 68px;
font-size: 30px;
text-align: left;
background-color: rgba(5, 42, 71, 1);
color: rgba(116, 196, 252, 1);
padding-top: 14px;
padding-right: 13px;
padding-bottom: 14px;
padding-left: 13px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.ui-popup .ui-popup-content {
width: 100%;
min-height: 14rem;
font-size: 2.125rem;
text-align: left;
color: rgba(245, 245, 245, 1);
background-color: rgba(0, 17, 33, 1);
padding: 10px 16px 9px 16px;
overflow: auto;
-webkit-overflow-scrolling: touch;
-moz-overflow-scrolling: touch;
-o-overflow-scrolling: touch;
-ms-overflow-scrolling: touch;
overflow-scrolling: touch;
}
.ui-popup .ui-popup-content .ui-listview,
.ui-popup .ui-popup-content .ui-inline-listview {
margin: -0.5625rem -1rem -0.5625rem -1rem;
}
.ui-popup .ui-popup-content .ui-listview li,
.ui-popup .ui-popup-content .ui-inline-listview li {
font-size: 2.125rem;
}
.ui-popup .ui-popup-footer {
width: 100%;
height: 83px;
}
.ui-popup .ui-popup-footer .ui-btn {
padding-top: 27px;
padding-bottom: 28px;
font-size: 30px;
}
.ui-popup .ui-popup-footer .ui-btn.ui-btn-icon-only {
position: relative;
text-indent: -624.9375rem;
}
.ui-popup .ui-popup-footer .ui-btn.ui-btn-icon-only::before {
content: "";
position: absolute;
-webkit-mask-size: 3.125rem 3.125rem;
-webkit-mask-repeat: no-repeat;
width: 3.125rem;
height: 3.125rem;
top: 50%;
left: 50%;
-webkit-transform: translate3d(-50%, -50%, 0);
}
.ui-popup.ui-popup-active {
display: block;
}
.ui-popup.in {
display: block;
}
.ui-popup.ui-build {
display: block;
visibility: hidden;
}
/***********************
Popup Container Scroll
***********************/
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) {
max-height: 100%;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-wrapper {
width: 100%;
max-height: 100%;
overflow: auto !important;
-webkit-overflow-scrolling: touch;
-moz-overflow-scrolling: touch;
-o-overflow-scrolling: touch;
-ms-overflow-scrolling: touch;
overflow-scrolling: touch;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-content {
height: auto !important;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-header.ui-fixed {
position: absolute !important;
top: 0;
left: 0;
width: 100%;
z-index: 2100;
}
.ui-popup.ui-scroll-on:not(.ui-ctxpopup):not(.ui-popup-toast) .ui-popup-footer.ui-fixed {
position: absolute !important;
bottom: 0;
left: 0;
width: 100%;
z-index: 2100;
-webkit-transform: translate3d(0, 0, 0);
}
/***********************
Option Popup
***********************/
.ui-popup.ui-ctxpopup {
background-color: transparent;
width: 100%;
border: 0 none;
border-radius: 0;
overflow: visible;
}
div.ui-popup.ui-ctxpopup.ui-popup-content-size {
width: auto;
padding: 0 0 0 0;
}
div.ui-popup.ui-ctxpopup.ui-popup-content-size .ui-popup-content {
width: auto;
}
.ui-popup.ui-ctxpopup .ui-popup-content {
background-color: rgba(245, 245, 245, 1);
border: 1px solid rgba(112, 112, 112, 0.9);
color: #333;
min-height: initial;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview li {
color: rgba(8, 8, 8, 1);
border-bottom: 1px solid rgba(204, 204, 204, 1);
min-height: 90px;
padding: 21px 8px 20px 16px;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview li a {
color: rgba(8, 8, 8, 1);
margin: -21px -16px;
padding: 21px 16px;
min-height: 48px;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview li a:active {
color: rgba(245, 245, 245, 1);
background-color: rgba(19, 99, 145, 1);
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview li:last-child {
border: none;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview .ui-list-icon {
position: relative;
padding-left: 4.25rem;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview .ui-list-icon a {
margin-left: -4.25rem;
padding-left: 4.25rem;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview .ui-list-icon::before {
content: "";
position: absolute;
top: 50%;
left: 0.375rem;
width: 3.5rem;
height: 3.5rem;
margin-top: -1.75rem;
background-color: rgba(99, 99, 99, 0.9);
-webkit-mask-size: 3.5rem 3.5rem;
-moz-mask-size: 3.5rem 3.5rem;
-ms-mask-size: 3.5rem 3.5rem;
-o-mask-size: 3.5rem 3.5rem;
mask-size: 3.5rem 3.5rem;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-listview a.ui-list-icon:active::before {
background-color: rgba(245, 245, 245, 1);
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview {
height: 4.3125rem;
overflow: hidden;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview li {
float: left;
height: 4.1875rem;
line-height: 4.1875rem;
border-right: 1px solid rgba(8, 8, 8, 1);
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview li:last-child {
border: none;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview li a {
display: block;
width: 100%;
height: 100%;
padding: 0rem 0.5rem;
color: rgba(8, 8, 8, 1);
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview li a:active {
color: rgba(245, 245, 245, 1);
background-color: rgba(19, 99, 145, 1);
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview .ui-list-icon-notext {
width: 3.6875rem;
height: 4.1875rem;
text-indent: -1000em;
overflow: hidden;
position: relative;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview .ui-list-icon-notext::before {
content: "";
position: absolute;
top: 50%;
left: 50%;
width: 2.6875rem;
height: 2.6875rem;
margin-top: -1.34375rem;
margin-left: -1.34375rem;
background-color: rgba(8, 8, 8, 1);
-webkit-mask-size: 2.6875rem 2.6875rem;
-moz-mask-size: 2.6875rem 2.6875rem;
-ms-mask-size: 2.6875rem 2.6875rem;
-o-mask-size: 2.6875rem 2.6875rem;
mask-size: 2.6875rem 2.6875rem;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-popup.ui-ctxpopup .ui-popup-content .ui-inline-listview a.ui-list-icon-notext:active::before {
background-color: rgba(245, 245, 245, 1);
}
.ui-popup.ui-ctxpopup .ui-arrow {
width: 36px;
height: 15px;
overflow: hidden;
position: absolute;
display: none;
-webkit-animation-duration: 0;
-moz-animation-duration: 0;
-ms-animation-duration: 0;
-o-animation-duration: 0;
}
.ui-popup.ui-ctxpopup .ui-arrow span {
width: 20px;
height: 20px;
background-color: rgba(245, 245, 245, 1);
border: 1px solid rgba(112, 112, 112, 0.9);
position: absolute;
top: 6px;
left: 8px;
-webkit-transform: rotate(55deg) skew(24deg);
-moz-transform: rotate(55deg) skew(24deg);
-ms-transform: rotate(55deg) skew(24deg);
-o-transform: rotate(55deg) skew(24deg);
transform: rotate(55deg) skew(24deg);
-webkit-animation-duration: 0;
-moz-animation-duration: 0;
-ms-animation-duration: 0;
-o-animation-duration: 0;
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-b {
padding: 20px 20px 0 20px;
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-t {
padding: 0 20px 20px 20px;
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-r {
padding: 20px 0 20px 20px;
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-l {
padding: 20px 20px 20px 0;
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-b .ui-arrow {
display: block;
bottom: -14px;
-webkit-transform: rotate(180deg);
-moz-transform: rotate(180deg);
-ms-transform: rotate(180deg);
-o-transform: rotate(180deg);
transform: rotate(180deg);
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-t .ui-arrow {
display: block;
top: -14px;
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-r .ui-arrow {
display: block;
right: -25px;
-webkit-transform: rotate(90deg);
-moz-transform: rotate(90deg);
-ms-transform: rotate(90deg);
-o-transform: rotate(90deg);
transform: rotate(90deg);
}
.ui-popup.ui-ctxpopup.ui-popup-arrow-l .ui-arrow {
display: block;
left: -25px;
-webkit-transform: rotate(-90deg);
-moz-transform: rotate(-90deg);
-ms-transform: rotate(-90deg);
-o-transform: rotate(-90deg);
transform: rotate(-90deg);
}
/****************************************
Popup Toast
****************************************/
.ui-popup.ui-popup-toast {
margin-left: 0;
margin-right: 0;
margin-bottom: 0;
font-size: 1.5rem;
border: 3px solid rgba(0, 55, 92, 1);
border-radius: 0;
border-left: 0;
border-right: 0;
position: absolute;
bottom: 0;
top: auto !important;
height: auto;
}
.ui-popup.ui-popup-toast .ui-popup-content {
min-height: initial;
padding: 0.6875rem 2.625rem 0.75rem 2.625rem;
background-color: rgba(0, 17, 33, 1);
}
.ui-popup.ui-popup-toast.ui-popup-toast-graphic .ui-popup-toast-icon {
width: 3.5rem;
height: 3.5rem;
position: relative;
left: 50%;
transform: translateX(-50%);
text-indent: -10000px;
overflow: hidden;
}
.ui-popup.ui-popup-toast.ui-popup-toast-graphic .ui-popup-toast-icon::after {
content: "";
width: 100%;
height: 100%;
position: absolute;
left: 0;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-position: 50% 50%;
-moz-mask-position: 50% 50%;
-ms-mask-position: 50% 50%;
-o-mask-position: 50% 50%;
mask-position: 50% 50%;
}
/****************************************
Popup Overlay
****************************************/
.ui-popup-overlay {
position: fixed;
display: none;
top: 0;
width: 100%;
height: 100%;
z-index: 100;
background-color: rgba(8, 8, 8, 0.7);
}
.ui-popup-overlay.in {
display: block;
}
/****************************************
Popup Transition
****************************************/
.ui-popup.slideup.in {
-webkit-animation-name: popupslideinfrombottom;
-moz-animation-name: popupslideinfrombottom;
-ms-animation-name: popupslideinfrombottom;
-o-animation-name: popupslideinfrombottom;
animation-name: popupslideinfrombottom;
-webkit-animation-duration: 250ms;
-moz-animation-duration: 250ms;
-ms-animation-duration: 250ms;
-o-animation-duration: 250ms;
-webkit-transition-timing-function: ease-in-out;
-moz-transition-timing-function: ease-in-out;
-o-transition-timing-function: ease-in-out;
-ms-transition-timing-function: ease-in-out;
transition-timing-function: ease-in-out;
}
.ui-popup.slideup.out {
-webkit-animation-name: popupslideouttobottom;
-moz-animation-name: popupslideouttobottom;
-ms-animation-name: popupslideouttobottom;
-o-animation-name: popupslideouttobottom;
animation-name: popupslideouttobottom;
-webkit-animation-duration: 250ms;
-moz-animation-duration: 250ms;
-ms-animation-duration: 250ms;
-o-animation-duration: 250ms;
-webkit-animation-fill-mode: forwards;
animation-fill-mode: forwards;
-webkit-transition-timing-function: ease-in-out;
-moz-transition-timing-function: ease-in-out;
-o-transition-timing-function: ease-in-out;
-ms-transition-timing-function: ease-in-out;
transition-timing-function: ease-in-out;
}
@-webkit-keyframes popupslideinfrombottom {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
}
@-moz-keyframes popupslideinfrombottom {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
}
@-ms-keyframes popupslideinfrombottom {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
}
@-o-keyframes popupslideinfrombottom {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
}
@keyframes popupslideinfrombottom {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
}
@-webkit-keyframes popupslideouttobottom {
from {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@-moz-keyframes popupslideouttobottom {
from {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@-ms-keyframes popupslideouttobottom {
from {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@-o-keyframes popupslideouttobottom {
from {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@keyframes popupslideouttobottom {
from {
-webkit-transform: translateY(0%);
-moz-transform: translateY(0%);
-ms-transform: translateY(0%);
-o-transform: translateY(0%);
transform: translateY(0%);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
.ui-popup.pop.in {
-webkit-animation-name: popuppopin;
-webkit-animation-duration: 350ms;
-webkit-transition-timing-function: ease-in-out;
animation-name: popuppopin;
animation-duration: 350ms;
transition-timing-function: ease-in-out;
}
.ui-popup.pop.out {
-webkit-animation-name: popuppopout;
-webkit-animation-duration: 350ms;
-webkit-transition-timing-function: ease-in-out;
animation-name: popuppopout;
animation-duration: 350ms;
transition-timing-function: ease-in-out;
}
@-webkit-keyframes popuppopin {
from {
transform: scale3d(1.35, 1.35, 1);
opacity: 0;
}
to {
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@-webkit-keyframes popuppopout {
from {
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
transform: scale3d(1.35, 1.35, 1);
opacity: 0;
}
}
/****************************************
Toast Popup Transition
****************************************/
.ui-popup.ui-popup-toast.pop.in {
-webkit-animation-name: toastpopuppopin !important;
-webkit-animation-duration: 250ms;
-webkit-transition-timing-function: ease-in-out;
animation-name: toastpopuppopin !important;
animation-duration: 250ms;
transition-timing-function: ease-in-out;
}
.ui-popup.ui-popup-toast.pop.out {
-webkit-animation-name: toastpopuppopout !important;
-webkit-animation-duration: 250ms;
-webkit-transition-timing-function: ease-in-out;
-webkit-animation-fill-mode: forwards;
animation-name: toastpopuppopout !important;
animation-duration: 250ms;
animation-fill-mode: forwards;
transition-timing-function: ease-in-out;
}
@-webkit-keyframes toastpopuppopin {
from {
-webkit-transform: translateY(100%);
opacity: 0;
}
to {
-webkit-transform: translateY(0%);
opacity: 1;
}
}
@-webkit-keyframes toastpopuppopout {
from {
-webkit-transform: translateY(0%);
opacity: 1;
}
to {
-webkit-transform: translateY(100%);
opacity: 0;
}
}
/****************************************
Popup Overlay Transition
****************************************/
.ui-popup-overlay.slideup.in,
.ui-popup-overlay.pop.in {
-webkit-animation: fadein 250ms;
-moz-animation: fadein 250ms;
-ms-animation: fadein 250ms;
-o-animation: fadein 250ms;
animation: fadein 250ms;
}
.ui-popup-overlay.slideup.out,
.ui-popup-overlay.pop.out {
-webkit-animation: fadeout 250ms forwards;
-moz-animation: fadeout 250ms forwards;
-ms-animation: fadeout 250ms forwards;
-o-animation: fadeout 250ms forwards;
animation: fadeout 250ms forwards;
}
/****************************************
Option Popup Transition
****************************************/
.ui-popup.ui-popup-arrow-t.slideup.in,
.ui-popup.ui-popup-arrow-l.slideup.in,
.ui-popup.ui-popup-arrow-r.slideup.in {
-webkit-animation: ctxpopupslideinfrombottom 250ms ease-in-out;
-moz-animation: ctxpopupslideinfrombottom 250ms ease-in-out;
-ms-animation: ctxpopupslideinfrombottom 250ms ease-in-out;
-o-animation: ctxpopupslideinfrombottom 250ms ease-in-out;
animation: ctxpopupslideinfrombottom 250ms ease-in-out;
}
.ui-popup.ui-popup-arrow-t.slideup.out,
.ui-popup.ui-popup-arrow-l.slideup.out,
.ui-popup.ui-popup-arrow-r.slideup.out {
-webkit-animation: ctxpopupslideouttobottom 250ms ease-in-out;
-moz-animation: ctxpopupslideouttobottom 250ms ease-in-out;
-ms-animation: ctxpopupslideouttobottom 250ms ease-in-out;
-o-animation: ctxpopupslideouttobottom 250ms ease-in-out;
animation: ctxpopupslideouttobottom 250ms ease-in-out;
}
.ui-popup.ui-popup-arrow-b.slideup.in {
-webkit-animation: ctxpopupslideinfromtop 250ms ease-in-out;
-moz-animation: ctxpopupslideinfromtop 250ms ease-in-out;
-ms-animation: ctxpopupslideinfromtop 250ms ease-in-out;
-o-animation: ctxpopupslideinfromtop 250ms ease-in-out;
animation: ctxpopupslideinfromtop 250ms ease-in-out;
}
.ui-popup.ui-popup-arrow-b.slideup.out {
-webkit-animation: ctxpopupslideouttotop 250ms ease-in-out;
-moz-animation: ctxpopupslideouttotop 250ms ease-in-out;
-ms-animation: ctxpopupslideouttotop 250ms ease-in-out;
-o-animation: ctxpopupslideouttotop 250ms ease-in-out;
animation: ctxpopupslideouttotop 250ms ease-in-out;
}
@-webkit-keyframes ctxpopupslideinfrombottom {
from {
transform: translateY(20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-moz-keyframes ctxpopupslideinfrombottom {
from {
transform: translateY(20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-ms-keyframes ctxpopupslideinfrombottom {
from {
transform: translateY(20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-o-keyframes ctxpopupslideinfrombottom {
from {
transform: translateY(20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@keyframes ctxpopupslideinfrombottom {
from {
transform: translateY(20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-webkit-keyframes ctxpopupslideouttobottom {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(20%);
opacity: 0.0;
}
}
@-moz-keyframes ctxpopupslideouttobottom {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(20%);
opacity: 0.0;
}
}
@-ms-keyframes ctxpopupslideouttobottom {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(20%);
opacity: 0.0;
}
}
@-o-keyframes ctxpopupslideouttobottom {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(20%);
opacity: 0.0;
}
}
@keyframes ctxpopupslideouttobottom {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(20%);
opacity: 0.0;
}
}
@-webkit-keyframes ctxpopupslideinfromtop {
from {
transform: translateY(-20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-ms-keyframes ctxpopupslideinfromtop {
from {
transform: translateY(-20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-ms-keyframes ctxpopupslideinfromtop {
from {
transform: translateY(-20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-o-keyframes ctxpopupslideinfromtop {
from {
transform: translateY(-20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@keyframes ctxpopupslideinfromtop {
from {
transform: translateY(-20%);
opacity: 0.0;
}
to {
transform: translateY(0%);
opacity: 1.0;
}
}
@-webkit-keyframes ctxpopupslideouttotop {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(-20%);
opacity: 0.0;
}
}
@-moz-keyframes ctxpopupslideouttotop {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(-20%);
opacity: 0.0;
}
}
@-ms-keyframes ctxpopupslideouttotop {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(-20%);
opacity: 0.0;
}
}
@-o-keyframes ctxpopupslideouttotop {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(-20%);
opacity: 0.0;
}
}
@keyframes ctxpopupslideouttotop {
from {
transform: translateY(0%);
opacity: 1.0;
}
to {
transform: translateY(-20%);
opacity: 0.0;
}
}
/************************
Listview
*************************/
.ui-listview li {
font-size: 2.5rem;
line-height: 3rem;
width: 100%;
min-height: 6.25rem;
padding: 1.625rem 0.5rem 1.625rem 1rem;
border-bottom: 1px solid rgba(56, 56, 56, 1);
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
position: relative;
color: rgba(245, 245, 245, 1);
}
.ui-listview li > * {
white-space: nowrap;
text-overflow: ellipsis;
overflow: hidden;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
-o-user-select: none;
user-select: none;
color: rgba(245, 245, 245, 1);
}
.ui-listview li > a {
display: block;
width: 100%;
height: 100%;
margin: -1.625rem -1rem;
padding: 1.625rem 0.5rem 1.625rem 1rem;
-o-box-sizing: content-box;
-ms-box-sizing: content-box;
-moz-box-sizing: content-box;
-webkit-box-sizing: content-box;
box-sizing: content-box;
}
.ui-listview li.ui-li-active {
background-color: rgba(8, 56, 94, 1);
color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-listview-divider {
height: 1.875rem;
min-height: 1.875rem;
line-height: 1.875rem;
padding: 0 0.5rem 0 1rem;
background-color: rgba(29, 72, 105, 1);
color: rgba(151, 197, 230, 1);
font-size: 1.5rem;
}
.ui-listview .ui-li-sub-text {
color: rgba(85, 135, 171, 1);
font-size: 1.75rem;
}
.ui-listview .ui-li-has-action-icon .ui-action-text {
width: calc(100% - 4.8125rem);
height: 100%;
}
.ui-listview .ui-li-has-action-icon .ui-action-text::after {
content: "";
position: absolute;
height: 3.5rem;
width: 0.125rem;
background-color: rgba(245, 245, 245, 1);
right: 4.25rem;
top: 20%;
}
.ui-listview .ui-li-has-action-icon .ui-action-divider {
display: none;
}
.ui-listview .ui-li-has-action-icon .ui-action-delete,
.ui-listview .ui-li-has-action-icon .ui-action-setting,
.ui-listview .ui-li-has-action-icon .ui-action-add {
position: absolute;
height: 100%;
width: 4.5rem;
right: 0;
top: 0;
-webkit-mask-size: 3.5rem 3.5rem;
-webkit-mask-repeat: no-repeat;
-webkit-mask-position: center center;
padding: 0;
margin: 0;
}
.ui-listview .ui-li-has-action-icon .ui-action-delete {
-webkit-mask-image: url(./images/listview/tw_list_delete_holo_dark.png);
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-setting {
-webkit-mask-image: url(./images/listview/tw_list_setting_holo_dark.png);
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-add {
-webkit-mask-image: url(./images/listview/tw_list_add_holo_dark.png);
background-color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon.ui-li-active {
background-color: transparent;
}
.ui-listview .ui-li-has-action-icon.ui-li-active .ui-action-text:active {
background-color: rgba(8, 56, 94, 1);
color: rgba(245, 245, 245, 1);
}
.ui-listview .ui-li-has-action-icon .ui-action-delete:active,
.ui-listview .ui-li-has-action-icon .ui-action-setting:active,
.ui-listview .ui-li-has-action-icon .ui-action-add:active {
background-color: rgba(8, 56, 94, 1);
color: rgba(245, 245, 245, 1);
}
.ui-listview .li-divider {
height: 1.875rem;
min-height: 1.875rem;
line-height: 1.875rem;
padding: 0 0.5rem 0 1rem;
background-color: rgba(29, 72, 105, 1);
color: rgba(151, 197, 230, 1);
font-size: 1.5rem;
}
.ui-datetime-widget {
position: relative;
height: 100%;
}
.ui-datetime {
text-align: center;
padding-top: 56px;
}
.ui-time-picker .ui-datetime {
padding-top: 53px;
}
.ui-datetime:only-child {
padding-top: 92px;
}
.ui-time-picker .ui-datetime:only-child {
padding-top: 89px;
}
.ui-datetime input[type='date'],
.ui-datetime input[type='datetime'],
.ui-datetime input[type='time'] {
display: none;
}
.ui-datefield {
height: 52px;
display: inline-block;
vertical-align: top;
}
.ui-datefield > .ui-btn {
float: left;
height: 52px;
padding: 0;
line-height: 52px;
font-size: 42px;
font-weight: 600;
border-radius: 2px;
background-color: rgba(255, 255, 255, 0.15);
}
.ui-datefield .ui-datefield-separator {
display: block;
height: 100%;
float: left;
width: 18px;
background: transparent url('./images/Datetimepicker/tw_timepicker_num_dot.png');
}
.ui-datefield > .ui-btn:enabled:focus {
background-color: rgba(255, 255, 255, 0.15);
}
.ui-datefield > .ui-btn:disabled {
color: #404040;
}
.ui-datafield > .ui-btn.ui-selected,
.ui-datefield > .ui-btn:enabled:active,
.ui-datefield > .ui-btn.ui-state-active:enabled,
.ui-datefield > .ui-btn.ui-selected:enabled {
background-color: #ff9000;
color: #000;
}
/* -- for datepicker -- */
.ui-date-picker .ui-datefield > .ui-btn:first-child {
margin-right: 2px;
}
.ui-date-picker .ui-datefield > .ui-btn:last-child {
margin-left: 2px;
}
.ui-date-picker .ui-datefield-year {
width: 114px;
}
.ui-date-picker .ui-datefield-month {
width: 102px;
}
.ui-date-picker .ui-datefield-day {
width: 68px;
}
/* -- end (for datepicker) -- */
/* -- for timepicker -- */
.ui-time-picker .ui-datefield-hour {
width: 74px;
}
.ui-time-picker .ui-datefield-min {
width: 74px;
}
.ui-time-picker .ui-datefield-period {
width: 84px;
margin-left: 10px;
}
.ui-time-picker .ui-datefield {
height: 58px;
}
.ui-time-picker .ui-datefield > .ui-btn {
height: 58px;
line-height: 58px;
font-size: 50px;
}
.ui-time-picker .ui-datefield > .ui-btn.ui-datefield-period {
font-size: 48px;
}
/* -- (end) for timepicker -- */
.ui-datetime-wheel {
height: 68px;
margin: 0 10px;
margin-top: 30px;
}
.ui-time-picker .ui-datetime-wheel {
margin-top: 27px;
}
.ui-datetime-wheel .ui-btn {
width: 68px;
height: 68px;
float: left;
}
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus {
float: right;
background: transparent url('./images/Datetimepicker/tw_timepicker_dialbtn_plus_normal.png') center center no-repeat;
}
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus:active:hover,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus:enabled:active,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus:enabled:active:hover {
background: transparent url('./images/Datetimepicker/tw_timepicker_dialbtn_plus_pressed.png') center center no-repeat;
}
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus:disabled,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus:disabled:active:hover,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus.ui-state-disabled,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-plus.ui-state-disabled:active:hover {
background: transparent url('./images/Datetimepicker/tw_timepicker_dialbtn_plus_dim.png') center center no-repeat;
}
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus {
background: transparent url('./images/Datetimepicker/tw_timepicker_dialbtn_minus_normal.png') center center no-repeat;
}
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus:active:hover,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus:enabled:active,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus:enabled:active:hover {
background: transparent url('./images/Datetimepicker/tw_timepicker_dialbtn_minus_pressed.png') center center no-repeat;
}
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus:disabled,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus:disabled:active:hover,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus.ui-state-disabled,
.ui-datetime-wheel .ui-btn.ui-datetime-wheel-minus.ui-state-disabled:active:hover {
background: transparent url('./images/Datetimepicker/tw_timepicker_dialbtn_minus_dim.png') center center no-repeat;
}
.ui-datetime-wheel .ui-wheel {
display: block;
margin-left: 68px;
margin-right: 68px;
height: 68px;
padding: 10px 0;
background: url('./images/Datetimepicker/tw_timepicker_wheel_01.png') center center no-repeat;
}
.ui-datetime-btns {
position: absolute;
bottom: 0;
width: 100%;
height: 76px;
border-top: 2px solid #262626;
}
/* Button separator */
.ui-datetime-btns::before {
content: "";
height: 34px;
width: 1px;
background-color: #262626;
position: absolute;
left: 160px;
top: 20px;
}
.ui-datetime-btns .ui-btn {
height: 100%;
position: relative;
background-color: transparent;
}
.ui-datetime-btns .ui-btn:enabled:active,
.ui-datetime-btns .ui-btn:enabled:focus,
.ui-datetime-btns .ui-btn:enabled:active:focus {
background-color: transparent;
}
.ui-datetime-btns .ui-btn-left {
width: 160px;
float: left;
background: url('./images/Datetimepicker/tw_timepicker_left_softbtn.png') center center no-repeat;
}
.ui-datetime-btns .ui-btn-right {
width: 159px;
float: right;
background: url('./images/Datetimepicker/tw_timepicker_right_softbtn.png') center center no-repeat;
}
.ui-datetime-btns .ui-btn:disabled,
.ui-datetime-btns .ui-btn.ui-state-disabled {
opacity: 0.2;
}
/* ---------------- */
.ui-datetime-periods {
height: 102px;
margin-top: 7px;
display: inline-block;
}
.ui-datetime-periods .ui-btn,
.ui-datetime-periods .ui-datetime-periods-separator {
height: 100%;
float: left;
padding: 0;
background: transparent;
}
.ui-datetime-periods .ui-btn {
width: 102px;
font-size: 34px;
line-height: 102px;
}
.ui-datetime-periods .ui-btn:enabled:focus {
background: transparent;
}
.ui-datetime-periods .ui-btn:enabled:active,
.ui-datetime-periods .ui-btn:enabled:active:focus,
.ui-datetime-periods .ui-btn.ui-selected:enabled:active,
.ui-datetime-periods .ui-btn.ui-state-active:enabled:active {
color: inherit;
text-decoration: none;
background: url('./images/Datetimepicker/tw_timepicker_time_bg.png') center center no-repeat;
}
.ui-datetime-periods .ui-btn.ui-selected:enabled,
.ui-datetime-periods .ui-btn.ui-state-active:enabled {
color: #ff9000;
text-decoration: underline;
}
.ui-datetime-periods .ui-datetime-periods-separator {
width: 24px;
background: url('./images/Datetimepicker/tw_timepicker_num_dot_01.png') center center no-repeat;
}
.ui-datetime-wheel.ui-hidden,
.ui-datetime-periods.ui-hidden {
display: none;
}
/* Transitions originally inspired by those from jQtouch, nice work, folks */
.ui-viewport-transitioning,
.ui-viewport-transitioning .ui-page {
width: 100%;
height: 100%;
overflow: hidden;
-webkit-box-sizing: border-box;
-moz-box-sizing: border-box;
box-sizing: border-box;
}
.ui-pre-in {
z-index: 100;
visibility: visible;
display: flex;
display: -webkit-flex;
flex-direction: column;
-webkit-flex-direction: column;
top: 0px;
left: 0px;
}
.in {
-webkit-animation-timing-function: ease-out;
-moz-animation-timing-function: ease-out;
-ms-animation-timing-function: ease-out;
-o-animation-timing-function: ease-out;
animation-timing-function: ease-out;
-webkit-animation-duration: 350ms;
-moz-animation-duration: 350ms;
-ms-animation-duration: 350ms;
-o-animation-duration: 350ms;
}
.out {
-webkit-animation-timing-function: ease-in;
-moz-animation-timing-function: ease-in;
-ms-animation-timing-function: ease-in;
-o-animation-timing-function: ease-in;
animation-timing-function: ease-in;
-webkit-animation-duration: 225ms;
-moz-animation-duration: 225ms;
-ms-animation-duration: 225ms;
-o-animation-duration: 225ms;
}
@-webkit-keyframes fadein {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@-moz-keyframes fadein {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@-ms-keyframes fadein {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@keyframes fadein {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@-webkit-keyframes fadeout {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@-moz-keyframes fadeout {
from {
opacity: 1;
}
to {
opacity: 0;
}
}
@-ms-keyframes fadeout {
from {
opacity: 1;
}
to {
opacity: 0;
}
}
@-o-keyframes fadeout {
from {
opacity: 1;
}
to {
opacity: 0;
}
}
@keyframes fadeout {
from {
opacity: 1;
}
to {
opacity: 0;
}
}
.fade.out {
opacity: 0;
-webkit-animation: fadeout 125ms;
-moz-animation: fadeout 125ms;
-ms-animation: fadeout 125ms;
-o-animation: fadeout 125ms;
animation: fadeout 125ms;
}
.fade.in {
opacity: 1;
-webkit-animation: fadein 225ms;
-moz-animation: fadein 225ms;
-ms-animation: fadein 225ms;
-o-animation: fadein 225ms;
animation: fadein 225ms;
}
@-webkit-keyframes slideupin {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
}
@-o-keyframes slideupin {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
}
@-ms-keyframes slideupin {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
}
@-moz-keyframes slideupin {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
}
@keyframes slideupin {
from {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
to {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
}
@-webkit-keyframes slideupout {
from {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
}
@-o-keyframes slideupout {
from {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
}
@-ms-keyframes slideupout {
from {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
}
@-moz-keyframes slideupout {
from {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
}
@keyframes slideupout {
from {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
}
@-webkit-keyframes slideupinreverse {
from {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
to {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@-o-keyframes slideupinreverse {
from {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
to {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@-ms-keyframes slideupinreverse {
from {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
to {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@-moz-keyframes slideupinreverse {
from {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
to {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@keyframes slideupinreverse {
from {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
opacity: .3;
}
to {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@-webkit-keyframes slideupoutreverse {
from {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@-o-keyframes slideupoutreverse {
from {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@-ms-keyframes slideupoutreverse {
from {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@-moz-keyframes slideupoutreverse {
from {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
@keyframes slideupoutreverse {
from {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
}
to {
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
}
}
.slideup.in:not(.ui-popup):not(.ui-popup-overlay) {
-webkit-transform: translateY(0);
-moz-transform: translateY(0);
-ms-transform: translateY(0);
-o-transform: translateY(0);
transform: translateY(0);
-webkit-animation: slideupin 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-moz-animation: slideupin 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-ms-animation: slideupin 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-o-animation: slideupin 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
animation: slideupin 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
}
.slideup.out:not(.ui-popup):not(.ui-popup-overlay) {
-webkit-transform: scale3d(0.6, 0.6, 1);
-moz-transform: scale3d(0.6, 0.6, 1);
-ms-transform: scale3d(0.6, 0.6, 1);
-o-transform: scale3d(0.6, 0.6, 1);
transform: scale3d(0.6, 0.6, 1);
-webkit-animation: slideupout 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-moz-animation: slideupout 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-ms-animation: slideupout 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-o-animation: slideupout 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
animation: slideupout 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
}
.slideup.out.reverse:not(.ui-popup):not(.ui-popup-overlay) {
z-index: 101;
-webkit-transform: translateY(100%);
-moz-transform: translateY(100%);
-ms-transform: translateY(100%);
-o-transform: translateY(100%);
transform: translateY(100%);
-webkit-animation: slideupoutreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-moz-animation: slideupoutreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-ms-animation: slideupoutreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-o-animation: slideupoutreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
animation: slideupoutreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
}
.slideup.in.reverse:not(.ui-popup):not(.ui-popup-overlay) {
-webkit-transform: scale3d(1, 1, 1);
-moz-transform: scale3d(1, 1, 1);
-ms-transform: scale3d(1, 1, 1);
-o-transform: scale3d(1, 1, 1);
transform: scale3d(1, 1, 1);
-webkit-animation: slideupinreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-moz-animation: slideupinreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-ms-animation: slideupinreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
-o-animation: slideupinreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
animation: slideupinreverse 400ms cubic-bezier(0.25, 0.46, 0.45, 1);
}
@-webkit-keyframes popin {
from {
transform: scale3d(0.3, 0.3, 1);
opacity: 0;
}
to {
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
@-webkit-keyframes popout {
from {
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
transform: scale3d(1.7, 1.7, 1);
opacity: 0;
}
}
@-webkit-keyframes popoutreverse {
from {
transform: scale3d(1, 1, 1);
opacity: 1;
}
to {
transform: scale3d(0.3, 0.3, 1);
opacity: 0;
}
}
@-webkit-keyframes popinreverse {
from {
transform: scale3d(1.7, 1.7, 1);
opacity: 0;
}
to {
transform: scale3d(1, 1, 1);
opacity: 1;
}
}
.pop.out.reverse {
z-index: 100;
-webkit-animation-duration: 225ms;
-webkit-animation-name: popoutreverse;
-moz-animation-duration: 225ms;
-moz-animation-name: popoutreverse;
animation-duration: 225ms;
animation-name: popoutreverse;
-webkit-animation-fill-mode: forwards;
-moz-animation-fill-mode: forwards;
-ms-animation-fill-mode: forwards;
-o-animation-fill-mode: forwards;
animation-fill-mode: forwards;
}
.pop.in.reverse {
z-index: 200;
-webkit-animation-duration: 225ms;
-webkit-animation-name: popinreverse;
-moz-animation-duration: 225ms;
-moz-animation-name: popinreverse;
animation-duration: 225ms;
animation-name: popinreverse;
-webkit-animation-fill-mode: forwards;
-moz-animation-fill-mode: forwards;
-ms-animation-fill-mode: forwards;
-o-animation-fill-mode: forwards;
animation-fill-mode: forwards;
}
.pop.in {
-webkit-animation-duration: 225ms;
-webkit-animation-name: popin;
-moz-animation-duration: 225ms;
-moz-animation-name: popin;
animation-duration: 225ms;
animation-name: popin;
-webkit-animation-fill-mode: forwards;
-moz-animation-fill-mode: forwards;
-ms-animation-fill-mode: forwards;
-o-animation-fill-mode: forwards;
animation-fill-mode: forwards;
}
.pop.out {
z-index: 200;
-webkit-animation-duration: 225ms;
-webkit-animation-name: popout;
-moz-animation-duration: 225ms;
-moz-animation-name: popout;
animation-duration: 225ms;
animation-name: popout;
-webkit-animation-fill-mode: forwards;
-moz-animation-fill-mode: forwards;
-ms-animation-fill-mode: forwards;
-o-animation-fill-mode: forwards;
animation-fill-mode: forwards;
}
.ui-indexscrollbar {
display: block;
position: fixed;
right: 0;
top: 0;
width: 2.8125rem;
height: 100%;
padding-left: 0.125rem;
background-color: rgba(33, 33, 33, 1);
z-index: 10;
overflow: visible;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
-o-user-select: none;
user-select: none;
cursor: pointer;
}
.ui-indexscrollbar ul {
position: absolute;
width: 100%;
top: 0;
}
.ui-indexscrollbar ul li {
color: rgba(84, 84, 84, 1);
display: block;
width: 100%;
text-align: center;
font-size: 1.5rem;
height: 2.25rem;
}
.ui-indexscrollbar ul li.ui-state-selected {
background-color: rgba(43, 43, 43, 1);
color: rgba(55, 161, 237, 1);
}
.ui-indexscrollbar ul.ui-indexscrollbar-supplementary {
position: relative;
height: auto;
top: 0;
right: -2.5rem;
width: 100%;
}
.ui-indexscrollbar ul.ui-indexscrollbar-supplementary li {
background-color: rgba(43, 43, 43, 1);
color: rgba(84, 84, 84, 1);
}
.ui-indexscrollbar ul.ui-indexscrollbar-supplementary li.ui-state-selected {
background-color: rgba(43, 43, 43, 1);
color: rgba(55, 161, 237, 1);
}
.ui-indexscrollbar + .ui-listview li {
padding-right: 2.5rem;
}
.ui-indexscrollbar-indicator {
position: fixed;
top: 0;
left: 0;
z-index: 9;
display: none;
}
.ui-indexscrollbar-indicator > span {
width: 8.75rem;
height: 5.4375rem;
position: absolute;
display: block;
top: 50%;
left: 50%;
margin-left: -5.625rem;
margin-top: -2.6875rem;
line-height: 5.4375rem;
font-size: 4.375rem;
text-align: center;
background-color: rgba(0, 61, 107, 0.95);
color: rgba(250, 250, 250, 1);
}
.ui-indexscrollbar-indicator > span > span.ui-selected {
color: rgba(55, 161, 237, 1);
}
.ui-circularindexscrollbar {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
pointer-events: none;
}
.ui-circularindexscrollbar .ui-circularindexscrollbar-indicator {
position: absolute;
top: 50%;
left: 50%;
width: 9.875rem;
height: 9.875rem;
line-height: 9.875rem;
border-radius: 50%;
background-color: rgba(0, 61, 107, 0.95);
-webkit-transition: opacity 250ms ease-out;
-moz-transition: opacity 250ms ease-out;
-o-transition: opacity 250ms ease-out;
-ms-transition: opacity 250ms ease-out;
transition: opacity 250ms ease-out;
-webkit-transform: translate(-50%, -50%);
-moz-transform: translate(-50%, -50%);
-ms-transform: translate(-50%, -50%);
-o-transform: translate(-50%, -50%);
transform: translate(-50%, -50%);
opacity: 0;
}
.ui-circularindexscrollbar.ui-circularindexscrollbar-show {
pointer-events: auto;
}
.ui-circularindexscrollbar.ui-circularindexscrollbar-show .ui-circularindexscrollbar-indicator {
opacity: 1;
}
.ui-circularindexscrollbar .ui-circularindexscrollbar-indicator-text {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
text-align: center;
font-size: 4.375rem;
color: rgba(55, 161, 237, 1);
}
/***************************************************************************
Scrollbar Style
***************************************************************************/
.ui-scrollbar-bar-type {
position: absolute;
}
.ui-scrollbar-bar-type.ui-scrollbar-horizontal {
left: 0;
bottom: 0;
width: 100%;
height: 8px;
}
.ui-scrollbar-bar-type.ui-scrollbar-vertical {
right: 0;
top: 0;
width: 0.5rem;
height: 100%;
}
.ui-scrollbar-bar-type .ui-scrollbar-indicator {
position: absolute;
background-color: rgba(102, 102, 102, 1);
}
.ui-scrollbar-bar-type.ui-scrollbar-horizontal .ui-scrollbar-indicator {
height: 0.25rem;
}
.ui-scrollbar-bar-type.ui-scrollbar-vertical .ui-scrollbar-indicator {
width: 0.25rem;
}
/***************************************************************************
Scrollbar Bouncing Effect
***************************************************************************/
.ui-scrollbar-bouncing-effect {
display: none;
position: absolute;
background-repeat: no-repeat;
-webkit-animation-duration: 0.47s;
-moz-animation-duration: 0.47s;
-ms-animation-duration: 0.47s;
-o-animation-duration: 0.47s;
-webkit-animation-timing-function: step-start;
-moz-animation-timing-function: step-start;
-ms-animation-timing-function: step-start;
-o-animation-timing-function: step-start;
animation-timing-function: step-start;
-webkit-animation-fill-mode: forwards;
-moz-animation-fill-mode: forwards;
-ms-animation-fill-mode: forwards;
-o-animation-fill-mode: forwards;
animation-fill-mode: forwards;
background-color: rgba(55, 161, 237, 0.7);
}
.ui-scrollbar-bouncing-effect::before {
content: "";
position: absolute;
background-color: rgba(55, 161, 237, 1);
}
.ui-scrollbar-bouncing-effect.ui-top,
.ui-scrollbar-bouncing-effect.ui-bottom {
width: 100%;
height: 3.125rem;
left: 0;
-webkit-mask-size: 100% 3.125rem;
}
.ui-scrollbar-bouncing-effect.ui-top::before,
.ui-scrollbar-bouncing-effect.ui-bottom::before {
width: 100%;
height: 0.125rem;
-webkit-mask-size: 100% 0.125rem;
}
.ui-scrollbar-bouncing-effect.ui-left,
.ui-scrollbar-bouncing-effect.ui-right {
width: 3.125rem;
height: 100%;
top: 0;
-webkit-mask-size: 3.125rem 100%;
}
.ui-scrollbar-bouncing-effect.ui-left::before,
.ui-scrollbar-bouncing-effect.ui-right::before {
width: 0.0625rem;
height: 100%;
-webkit-mask-size: 0.0625rem 100%;
}
.ui-scrollbar-bouncing-effect.ui-top {
top: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_top_glow.png");
-moz-mask-image: url("./images/Scroller/bouncing_top_glow.png");
-ms-mask-image: url("./images/Scroller/bouncing_top_glow.png");
-o-mask-image: url("./images/Scroller/bouncing_top_glow.png");
mask-image: url("./images/Scroller/bouncing_top_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-top::before {
-webkit-mask-image: url("./images/Scroller/bouncing_top_edge.png");
-moz-mask-image: url("./images/Scroller/bouncing_top_edge.png");
-ms-mask-image: url("./images/Scroller/bouncing_top_edge.png");
-o-mask-image: url("./images/Scroller/bouncing_top_edge.png");
mask-image: url("./images/Scroller/bouncing_top_edge.png");
}
.ui-scrollbar-bouncing-effect.ui-bottom {
bottom: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_bottom_glow.png");
-moz-mask-image: url("./images/Scroller/bouncing_bottom_glow.png");
-ms-mask-image: url("./images/Scroller/bouncing_bottom_glow.png");
-o-mask-image: url("./images/Scroller/bouncing_bottom_glow.png");
mask-image: url("./images/Scroller/bouncing_bottom_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-bottom::before {
-webkit-mask-image: url("./images/Scroller/bouncing_bottom_edge.png");
-moz-mask-image: url("./images/Scroller/bouncing_bottom_edge.png");
-ms-mask-image: url("./images/Scroller/bouncing_bottom_edge.png");
-o-mask-image: url("./images/Scroller/bouncing_bottom_edge.png");
mask-image: url("./images/Scroller/bouncing_bottom_edge.png");
}
.ui-scrollbar-bouncing-effect.ui-left {
left: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_left_glow.png");
-moz-mask-image: url("./images/Scroller/bouncing_left_glow.png");
-ms-mask-image: url("./images/Scroller/bouncing_left_glow.png");
-o-mask-image: url("./images/Scroller/bouncing_left_glow.png");
mask-image: url("./images/Scroller/bouncing_left_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-left::before {
-webkit-mask-image: url("./images/Scroller/bouncing_left_edge.png");
-moz-mask-image: url("./images/Scroller/bouncing_left_edge.png");
-ms-mask-image: url("./images/Scroller/bouncing_left_edge.png");
-o-mask-image: url("./images/Scroller/bouncing_left_edge.png");
mask-image: url("./images/Scroller/bouncing_left_edge.png");
}
.ui-scrollbar-bouncing-effect.ui-right {
right: 0;
-webkit-mask-image: url("./images/Scroller/bouncing_right_glow.png");
-moz-mask-image: url("./images/Scroller/bouncing_right_glow.png");
-ms-mask-image: url("./images/Scroller/bouncing_right_glow.png");
-o-mask-image: url("./images/Scroller/bouncing_right_glow.png");
mask-image: url("./images/Scroller/bouncing_right_glow.png");
}
.ui-scrollbar-bouncing-effect.ui-right::before {
-webkit-mask-image: url("./images/Scroller/bouncing_right_edge.png");
-moz-mask-image: url("./images/Scroller/bouncing_right_edge.png");
-ms-mask-image: url("./images/Scroller/bouncing_right_edge.png");
-o-mask-image: url("./images/Scroller/bouncing_right_edge.png");
mask-image: url("./images/Scroller/bouncing_right_edge.png");
}
@-webkit-keyframes ui-bouncing-show {
0% {
opacity: 0;
}
10% {
opacity: 0.1;
}
20% {
opacity: 0.2;
}
30% {
opacity: 0.3;
}
40% {
opacity: 0.4;
}
50% {
opacity: 0.5;
}
60% {
opacity: 0.6;
}
70% {
opacity: 0.7;
}
80% {
opacity: 0.8;
}
90% {
opacity: 0.9;
}
100% {
opacity: 1;
}
}
@-webkit-keyframes ui-bouncing-hide {
0% {
opacity: 1;
}
10% {
opacity: 0.9;
}
20% {
opacity: 0.8;
}
30% {
opacity: 0.7;
}
40% {
opacity: 0.6;
}
50% {
opacity: 0.5;
}
60% {
opacity: 0.4;
}
70% {
opacity: 0.3;
}
80% {
opacity: 0.2;
}
90% {
opacity: 0.1;
}
100% {
opacity: 0;
}
}
.ui-scrollbar-bouncing-effect.ui-show {
display: block;
-webkit-animation-name: ui-bouncing-show;
-moz-animation-name: ui-bouncing-show;
-ms-animation-name: ui-bouncing-show;
-o-animation-name: ui-bouncing-show;
animation-name: ui-bouncing-show;
}
.ui-scrollbar-bouncing-effect.ui-hide {
display: block;
-webkit-animation-name: ui-bouncing-hide;
-moz-animation-name: ui-bouncing-hide;
-ms-animation-name: ui-bouncing-hide;
-o-animation-name: ui-bouncing-hide;
animation-name: ui-bouncing-hide;
}
.ui-swipelist {
position: absolute;
top: 0;
}
.ui-swipelist .ui-swipelist-icon {
width: 20%;
height: 100%;
-webkit-mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
}
.ui-swipelist .ui-swipelist-text {
top: 50%;
transform: translate3d(0, -50%, 0);
}
.ui-swipelist-left,
.ui-swipelist-right {
position: absolute;
display: none;
}
.ui-swipelist-left {
background-color: rgba(98, 168, 24, 1);
background-image: -webkit-gradient(linear, left top, left top, color-stop(0%, rgba(98, 168, 24, 1)), color-stop(0% rgba(8, 8, 8, 1)));
background-image: -webkit-linear-gradient(left, rgba(98, 168, 24, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: -moz-linear-gradient(left, rgba(98, 168, 24, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: -ms-linear-gradient(left, rgba(98, 168, 24, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: -o-linear-gradient(left, rgba(98, 168, 24, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: linear-gradient(left, rgba(98, 168, 24, 1) 0%, rgba(8, 8, 8, 1) 0%);
}
.ui-swipelist-right {
background-color: rgba(235, 164, 23, 1);
background-image: -webkit-gradient(linear, right top, right top, color-stop(0%, rgba(235, 164, 23, 1)), color-stop(0% rgba(8, 8, 8, 1)));
background-image: -webkit-linear-gradient(right, rgba(235, 164, 23, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: -moz-linear-gradient(right, rgba(235, 164, 23, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: -ms-linear-gradient(right, rgba(235, 164, 23, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: -o-linear-gradient(right, rgba(235, 164, 23, 1) 0%, rgba(8, 8, 8, 1) 0%);
background-image: linear-gradient(right, rgba(235, 164, 23, 1) 0%, rgba(8, 8, 8, 1) 0%);
}
.ui-swipelist-icon,
.ui-swipelist-text {
position: absolute;
}
.ui-swipelist-left .ui-swipelist-icon {
/*
Icon and text need to set position or url and so on.
We thought that swipelist's content style need to be implemented by developer wanted.
*/
margin-left: 2%;
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url('./images/Swipelist/b_logs_icon_body_btn_call_nor.png');
-moz-mask-image: url('./images/Swipelist/b_logs_icon_body_btn_call_nor.png');
-ms-mask-image: url('./images/Swipelist/b_logs_icon_body_btn_call_nor.png');
-o-mask-image: url('./images/Swipelist/b_logs_icon_body_btn_call_nor.png');
mask-image: url('./images/Swipelist/b_logs_icon_body_btn_call_nor.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-swipelist-left .ui-swipelist-text {
/*
Icon and text need to set position or url and so on.
We thought that swipelist's content style need to be implemented by developer wanted.
*/
margin-left: 40%;
}
.ui-swipelist-right .ui-swipelist-icon {
/*
Icon and text need to set position or url and so on.
We thought that swipelist's content style need to be implemented by developer wanted.
*/
margin-left: 80%;
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url('./images/Swipelist/b_logs_icon_actionbar_btn_msg_nor.png');
-moz-mask-image: url('./images/Swipelist/b_logs_icon_actionbar_btn_msg_nor.png');
-ms-mask-image: url('./images/Swipelist/b_logs_icon_actionbar_btn_msg_nor.png');
-o-mask-image: url('./images/Swipelist/b_logs_icon_actionbar_btn_msg_nor.png');
mask-image: url('./images/Swipelist/b_logs_icon_actionbar_btn_msg_nor.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
.ui-swipelist-right .ui-swipelist-text {
/*
Icon and text need to set position or url and so on.
We thought that swipelist's content style need to be implemented by developer wanted.
*/
margin-left: 20%;
}
/***************************************************************************
Tab Style
***************************************************************************/
.ui-tab-indicator {
position: relative;
height: 6px;
}
.ui-tab-indicator .ui-tab-item {
position: absolute;
top: 0;
left: 0;
display: block;
height: 6px;
background-color: rgba(76, 103, 125, 1);
}
.ui-tab-indicator .ui-tab-item.ui-tab-active {
background-color: rgba(55, 161, 237, 1);
}
input[type="checkbox"]:not(.ui-switch-input),
input[type="radio"] {
box-sizing: border-box;
padding: 0;
height: 56px;
width: 56px;
font-size: 32px;
-webkit-appearance: none;
-moz-appearance: none;
-ms-appearance: none;
-o-appearance: none;
appearance: none;
position: relative;
}
input[type="radio"] {
background-color: rgba(0, 14, 26, 1);
-webkit-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
}
input[type="radio"]::after {
content: "";
background-color: rgba(245, 245, 245, 0.3);
-webkit-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-o-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
width: 100%;
height: 100%;
position: absolute;
}
input[type="radio"]:checked::before {
content: "";
background-color: rgba(112, 186, 15, 1);
-webkit-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
width: 100%;
height: 100%;
position: absolute;
}
input[type="radio"]:active {
background-color: rgba(0, 50, 84, 0.4);
-webkit-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_radiobox_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
}
input[type="radio"]:active::after {
content: "";
background-color: rgba(245, 245, 245, 0.3);
-webkit-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-o-mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
mask-image: url('images/Controller_icon/tw_btn_radiobox_line.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
width: 100%;
height: 100%;
position: absolute;
}
input[type="radio"]:active:checked::before {
content: "";
background-color: rgba(245, 245, 245, 1);
-webkit-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_radio_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
width: 100%;
height: 100%;
position: absolute;
}
input[type="checkbox"]:not(.ui-switch-input) {
-webkit-mask-image: url('images/Controller_icon/tw_btn_checkbox_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_checkbox_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_checkbox_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_checkbox_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_checkbox_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
background-color: rgba(0, 14, 26, 1);
}
input[type="checkbox"]:not(.ui-switch-input)::after {
content: "";
background-color: rgba(245, 245, 245, 0.3);
-webkit-mask-image: url('images/Controller_icon/tw_btn_checkbox_line_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_checkbox_line_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_checkbox_line_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_checkbox_line_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_checkbox_line_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
width: 100%;
height: 100%;
position: absolute;
}
input[type="checkbox"]:not(.ui-switch-input):checked::before {
content: "";
-webkit-mask-image: url('images/Controller_icon/tw_btn_check_holo_dark.png');
-moz-mask-image: url('images/Controller_icon/tw_btn_check_holo_dark.png');
-ms-mask-image: url('images/Controller_icon/tw_btn_check_holo_dark.png');
-o-mask-image: url('images/Controller_icon/tw_btn_check_holo_dark.png');
mask-image: url('images/Controller_icon/tw_btn_check_holo_dark.png');
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
background-color: rgba(112, 186, 15, 1);
width: 100%;
height: 100%;
position: absolute;
}
input[type="checkbox"]:not(.ui-switch-input):active {
background-color: rgba(0, 50, 84, 0.4);
}
input[type="checkbox"]:not(.ui-switch-input):active::after {
content: "";
background-color: rgba(245, 245, 245, 0.3);
width: 100%;
height: 100%;
position: absolute;
}
input[type="checkbox"]:not(.ui-switch-input):active:checked::before {
content: "";
background-color: rgba(245, 245, 245, 1);
width: 100%;
height: 100%;
position: absolute;
}
.ui-marquee {
position: relative;
white-space: nowrap;
overflow: hidden;
-webkit-user-select: none;
}
.ui-marquee.ui-marquee-gradient {
text-overflow: clip;
-webkit-mask-image: -webkit-linear-gradient(left, #ffffff 0%, #ffffff 85%, transparent 100%);
}
.ui-marquee.ui-marquee-ellipsis .ui-marquee-content {
display: block;
overflow: hidden;
text-overflow: ellipsis;
}
.ui-marquee-content {
overflow: visible;
display: inline-block;
}
.ui-marquee-anim-running {
-webkit-animation-play-state: running;
}
.ui-marquee-anim-stopped {
-webkit-animation-play-state: paused;
}
/****************************************************
Page Indicator
*****************************************************/
.ui-page-indicator {
position: absolute;
left: 50%;
-webkit-transform: translate3d(-50%, 0, 0);
-moz-transform: translate3d(-50%, 0, 0);
-ms-transform: translate3d(-50%, 0, 0);
-o-transform: translate3d(-50%, 0, 0);
transform: translate3d(-50%, 0, 0);
white-space: nowrap;
max-width: 100%;
overflow: hidden;
}
.ui-page-indicator .ui-page-indicator-item {
position: relative;
display: inline-block;
width: 0.8125rem;
height: 0.8125rem;
-webkit-mask-image: -webkit-radial-gradient(#000000 0.25rem, transparent 0.375rem);
background-color: rgba(245, 245, 245, 0.4);
margin-right: 0.5625rem;
-webkit-transition-duration: 250ms;
-moz-transition-duration: 250ms;
-o-transition-duration: 250ms;
-ms-transition-duration: 250ms;
transition-duration: 250ms;
}
.ui-page-indicator .ui-page-indicator-item:last-child {
margin-right: 0;
}
.ui-page-indicator .ui-page-indicator-item.ui-page-indicator-active {
background-color: rgba(245, 245, 245, 1);
}
.ui-page-indicator.ui-page-indicator-circular {
top: 0;
left: 0;
width: 100vw;
height: 100vw;
-webkit-transform: none;
-moz-transform: none;
-ms-transform: none;
-o-transform: none;
transform: none;
pointer-events: none;
}
.ui-page-indicator.ui-page-indicator-circular .ui-page-indicator-item {
-webkit-transition: none;
-moz-transition: none;
-o-transition: none;
-ms-transition: none;
transition: none;
position: absolute;
top: 50%;
left: 50%;
margin-top: -0.3125rem;
margin-left: -0.3125rem;
-webkit-mask-image: none;
background-color: transparent;
margin-right: 0;
}
.ui-page-indicator.ui-page-indicator-circular .ui-page-indicator-item::before {
content: "";
position: absolute;
width: 100%;
height: 100%;
-webkit-mask-image: -webkit-radial-gradient(#000000 0.125rem, transparent 0.25rem);
background-color: rgba(245, 245, 245, 0.4);
-webkit-transition: 250ms transform ease-out;
-moz-transition: 250ms transform ease-out;
-o-transition: 250ms transform ease-out;
-ms-transition: 250ms transform ease-out;
transition: 250ms transform ease-out;
}
.ui-page-indicator.ui-page-indicator-circular .ui-page-indicator-item.ui-page-indicator-active {
background-color: transparent;
}
.ui-page-indicator.ui-page-indicator-circular .ui-page-indicator-item.ui-page-indicator-active::before {
background-color: rgba(245, 245, 245, 1);
-webkit-transform: scale3d(1.5, 1.5, 1.5);
-moz-transform: scale3d(1.5, 1.5, 1.5);
-ms-transform: scale3d(1.5, 1.5, 1.5);
-o-transform: scale3d(1.5, 1.5, 1.5);
transform: scale3d(1.5, 1.5, 1.5);
}
.ui-snap-container {
-webkit-scroll-snap-type: mandatory;
-webkit-scroll-snap-destination: 50% 50%;
}
.ui-snap-container.ui-snap-disabled {
-webkit-scroll-snap-type: none;
}
.ui-snap-listview-item {
-webkit-scroll-snap-coordinate: 50% 50%;
}
.ui-drawer {
width: 100%;
height: 100%;
position: fixed;
top: 0;
background-color: rgba(8, 8, 8, 1);
z-index: 1201;
box-sizing: border-box;
}
.ui-drawer.ui-drawer-close {
overflow: hidden;
}
.ui-drawer.ui-drawer-open {
overflow: auto;
}
.ui-drawer-overlay {
top: 0;
position: fixed;
background-color: rgba(0, 0, 0, 0.8);
z-index: 1200;
}
.ui-section-changer {
height: 100%;
}
.ui-view-switcher {
height: 100%;
overflow: hidden;
}
.ui-view-switcher.ui-view-carousel .ui-view {
display: none;
position: absolute;
z-index: -1000;
top: 50%;
background-color: #fafafa;
}
.ui-view-switcher.ui-view-carousel .ui-view.ui-view-carousel-active {
display: inline;
z-index: 100;
left: 50%;
transform: translate3d(-50%, 0, 0);
}
.ui-view-switcher.ui-view-carousel .ui-view.ui-view-carousel-active .ui-view-carousel-dim {
opacity: 0;
}
.ui-view-switcher.ui-view-carousel .ui-view.ui-view-carousel-left {
display: inline;
left: 0;
}
.ui-view-switcher.ui-view-carousel .ui-view.ui-view-carousel-right {
display: inline;
right: 0;
}
.ui-view-switcher.ui-view-carousel .ui-view .ui-view-carousel-dim {
position: absolute;
top: 0;
width: 100%;
height: 100%;
opacity: 0.6;
background-color: #000000;
}
.ui-slider {
position: relative;
box-sizing: border-box;
background-color: rgba(71, 71, 71, 1);
}
.ui-slider .ui-slider-value {
background-color: rgba(55, 161, 237, 1);
}
.ui-slider .ui-slider-handler {
width: 2.25rem;
height: 2.25rem;
background-color: #fafafa;
border-radius: 50%;
position: absolute;
text-indent: 9999px;
-webkit-transition: transform 300ms ease-out;
-moz-transition: transform 300ms ease-out;
-o-transition: transform 300ms ease-out;
-ms-transition: transform 300ms ease-out;
transition: transform 300ms ease-out;
}
.ui-slider.ui-slider-horizontal {
width: 100%;
height: 6px;
margin: 20px 0;
}
.ui-slider.ui-slider-horizontal .ui-slider-value {
height: 6px;
}
.ui-slider.ui-slider-horizontal .ui-slider-handler {
top: 50%;
margin-left: -20px;
border: 3px solid rgba(55, 161, 237, 1);
-webkit-transform: translate(0, -50%) scale(0.6);
-moz-transform: translate(0, -50%) scale(0.6);
-ms-transform: translate(0, -50%) scale(0.6);
-o-transform: translate(0, -50%) scale(0.6);
transform: translate(0, -50%) scale(0.6);
}
.ui-slider.ui-slider-horizontal .ui-slider-handler.ui-slider-handler-active {
-webkit-transform: translate(0, -50%) scale(1);
-moz-transform: translate(0, -50%) scale(1);
-ms-transform: translate(0, -50%) scale(1);
-o-transform: translate(0, -50%) scale(1);
transform: translate(0, -50%) scale(1);
}
.ui-selector {
position: absolute;
top: 0;
left: 0;
height: 100%;
width: 100%;
background-color: rgba(8, 8, 8, 0.7);
}
.ui-selector .ui-layer {
position: absolute;
display: none;
width: 3.375rem;
height: 3.375rem;
border-radius: 50%;
}
.ui-selector .ui-layer.ui-layer-active {
display: block;
width: 100%;
height: 100%;
z-index: 3;
}
.ui-selector .ui-layer.ui-layer-active .ui-item {
visibility: visible;
-webkit-transition: transform 300ms ease-out;
-moz-transition: transform 300ms ease-out;
-o-transition: transform 300ms ease-out;
-ms-transition: transform 300ms ease-out;
transition: transform 300ms ease-out;
}
.ui-selector .ui-layer.ui-layer-hide .ui-item {
visibility: visible;
-webkit-transition: transform 100ms ease-out;
-moz-transition: transform 100ms ease-out;
-o-transition: transform 100ms ease-out;
-ms-transition: transform 100ms ease-out;
transition: transform 100ms ease-out;
}
.ui-selector .ui-layer.ui-layer-prev {
display: block;
top: 50%;
left: 50%;
margin: -1.75rem;
z-index: 2;
background-color: rgba(102, 102, 102, 1);
}
.ui-selector .ui-layer.ui-layer-prev-prev {
display: block;
top: 50%;
left: 50%;
margin: -1.75rem;
z-index: 1;
background-color: rgba(51, 51, 51, 1);
}
.ui-selector .ui-layer.ui-layer-next {
display: block;
top: 50%;
left: 50%;
margin: -1.75rem;
z-index: 2;
background-color: rgba(102, 102, 102, 1);
}
.ui-selector .ui-layer.ui-layer-next-next {
display: block;
top: 50%;
left: 50%;
margin: -1.75rem;
z-index: 1;
background-color: rgba(51, 51, 51, 1);
}
.ui-selector .ui-layer .ui-item {
background-color: rgba(245, 245, 245, 1);
position: absolute;
visibility: hidden;
width: 3.5rem;
height: 3.5rem;
top: 50%;
left: 50%;
border-radius: 50%;
margin: -1.75rem;
z-index: 100;
}
.ui-selector .ui-layer .ui-item:active {
background-color: rgba(186, 186, 186, 1);
}
.ui-selector .ui-layer .ui-item::before {
-webkit-transition: background-color 200ms ease-out;
}
.ui-selector .ui-selector-indicator,
.ui-selector .ui-selector-indicator-arrow {
position: absolute;
width: 13.125rem;
height: 13.125rem;
top: 50%;
left: 50%;
margin: -6.5625rem;
text-align: center;
line-height: 13.125rem;
font-size: 1.75rem;
}
.ui-selector .ui-selector-indicator {
z-index: 4;
width: 10rem;
height: 10rem;
margin: -5rem;
line-height: 10rem;
}
.ui-selector .ui-selector-indicator::before {
content: "";
position: absolute;
width: 100%;
height: 100%;
left: 0;
top: 0;
border-radius: 50%;
background-color: transparent;
-webkit-transform: scale(0.8);
-moz-transform: scale(0.8);
-ms-transform: scale(0.8);
-o-transform: scale(0.8);
transform: scale(0.8);
-webkit-transition: -webkit-transform 250ms;
-moz-transition: -webkit-transform 250ms;
-o-transition: -webkit-transform 250ms;
-ms-transition: -webkit-transform 250ms;
transition: -webkit-transform 250ms;
}
.ui-selector .ui-selector-indicator:active::before {
z-index: -1;
background-color: rgba(33, 41, 47, 0.5);
-webkit-transform: scale(1);
-moz-transform: scale(1);
-ms-transform: scale(1);
-o-transform: scale(1);
transform: scale(1);
}
.ui-selector .ui-selector-indicator-arrow {
z-index: 0;
}
.ui-selector .ui-selector-indicator-arrow::before {
position: absolute;
content: "";
width: 100%;
height: 100%;
top: 0;
left: 0;
-webkit-mask-image: url('./images/Common/b_rotary_selector_center_pointer.png');
-moz-mask-image: url('./images/Common/b_rotary_selector_center_pointer.png');
-ms-mask-image: url('./images/Common/b_rotary_selector_center_pointer.png');
-o-mask-image: url('./images/Common/b_rotary_selector_center_pointer.png');
mask-image: url('./images/Common/b_rotary_selector_center_pointer.png');
-webkit-mask-size: 100% 100%;
-moz-mask-size: 100% 100%;
-ms-mask-size: 100% 100%;
-o-mask-size: 100% 100%;
mask-size: 100% 100%;
-webkit-mask-repeat: no-repeat;
-moz-mask-repeat: no-repeat;
-ms-mask-repeat: no-repeat;
-o-mask-repeat: no-repeat;
mask-repeat: no-repeat;
background-color: #0096ff;
}
.tau-info-theme:after {
content: "default";
}
This file has been truncated, but you can view the full file.
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
(function(window, document, undefined) {
'use strict';
var ns = window.tau = window.tau || {},
nsConfig = window.tauConfig = window.tauConfig || {};
nsConfig.rootNamespace = 'tau';
nsConfig.fileName = 'tau';
ns.version = '0.11.7';
/*global window, console, define, ns, nsConfig */
/*jslint plusplus:true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Core namespace
* Object contains main framework methods.
* @class ns
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
* @author Tomasz Lukawski <t.lukawski@samsung.com>
*/
(function (document, ns, nsConfig) {
"use strict";
var idNumberCounter = 0,
currentDate = +new Date(),
slice = [].slice,
rootNamespace = nsConfig.rootNamespace,
fileName = nsConfig.fileName,
infoForLog = function (args) {
var dateNow = new Date();
args.unshift('[' + rootNamespace + '][' + dateNow.toLocaleString() + ']');
};
/**
* Return unique id
* @method getUniqueId
* @static
* @return {string}
* @member ns
*/
ns.getUniqueId = function () {
return rootNamespace + "-" + ns.getNumberUniqueId() + "-" + currentDate;
};
/**
* Return unique id
* @method getNumberUniqueId
* @static
* @return {number}
* @member ns
*/
ns.getNumberUniqueId = function () {
return idNumberCounter++;
};
/**
* logs supplied messages/arguments
* @method log
* @static
* @param {...*} argument
* @member ns
*/
ns.log = function () {
var args = slice.call(arguments);
infoForLog(args);
if (console) {
console.log.apply(console, args);
}
};
/**
* logs supplied messages/arguments ad marks it as warning
* @method warn
* @static
* @param {...*} argument
* @member ns
*/
ns.warn = function () {
var args = slice.call(arguments);
infoForLog(args);
if (console) {
console.warn.apply(console, args);
}
};
/**
* logs supplied messages/arguments and marks it as error
* @method error
* @static
* @param {...*} argument
* @member ns
*/
ns.error = function () {
var args = slice.call(arguments);
infoForLog(args);
if (console) {
console.error.apply(console, args);
}
};
/**
* get from nsConfig
* @method getConfig
* @param {string} key
* @param {*} defaultValue
* @return {*}
* @static
* @member ns
*/
ns.getConfig = function (key, defaultValue) {
return nsConfig[key] === undefined ? defaultValue : nsConfig[key];
};
/**
* set in nsConfig
* @method setConfig
* @param {string} key
* @param {*} value
* @param {boolean} [asDefault=false] value should be treated as default (doesn't overwrites the config[key] if it already exists)
* @static
* @member ns
*/
ns.setConfig = function (key, value, asDefault) {
if (!asDefault || (asDefault && nsConfig[key] === undefined)) {
nsConfig[key] = value;
}
};
/**
* Return path for framework script file.
* @method getFrameworkPath
* @returns {?string}
* @member ns
*/
ns.getFrameworkPath = function () {
var scripts = document.getElementsByTagName('script'),
countScripts = scripts.length,
i,
url,
arrayUrl,
count;
for (i = 0; i < countScripts; i++) {
url = scripts[i].src;
arrayUrl = url.split('/');
count = arrayUrl.length;
if (arrayUrl[count - 1] === fileName + '.js' || arrayUrl[count - 1] === fileName + '.min.js') {
return arrayUrl.slice(0, count - 1).join('/');
}
}
return null;
};
}(window.document, ns, nsConfig));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint plusplus: true, nomen: true */
// * @TODO add support of $.mobile.buttonMarkup.hoverDelay
/*
* Defaults settings object
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @class ns.defaults
*/
(function (ns) {
"use strict";
ns.defaults = {};
Object.defineProperty(ns.defaults, "autoInitializePage", {
get: function(){
return ns.getConfig("autoInitializePage", true);
},
set: function(value){
return ns.setConfig("autoInitializePage", value);
}
});
Object.defineProperty(ns.defaults, "dynamicBaseEnabled", {
get: function(){
return ns.getConfig("dynamicBaseEnabled", true);
},
set: function(value){
return ns.setConfig("dynamicBaseEnabled", value);
}
});
Object.defineProperty(ns.defaults, "pageTransition", {
get: function(){
return ns.getConfig("pageTransition", "none");
},
set: function(value){
return ns.setConfig("pageTransition", value);
}
});
Object.defineProperty(ns.defaults, "popupTransition", {
get: function(){
return ns.getConfig("popupTransition", "none");
},
set: function(value){
return ns.setConfig("popupTransition", value);
}
});
Object.defineProperty(ns.defaults, "popupFullSize", {
get: function(){
return ns.getConfig("popupFullSize", false);
},
set: function(value){
return ns.setConfig("popupFullSize", value);
}
});
Object.defineProperty(ns.defaults, "enablePageScroll", {
get: function(){
return ns.getConfig("enablePageScroll", false);
},
set: function(value){
return ns.setConfig("enablePageScroll", value);
}
});
Object.defineProperty(ns.defaults, "scrollEndEffectArea", {
get: function(){
return ns.getConfig("scrollEndEffectArea", "content");
},
set: function(value){
return ns.setConfig("scrollEndEffectArea", value);
}
});
Object.defineProperty(ns.defaults, "enablePopupScroll", {
get: function(){
return ns.getConfig("enablePopupScroll", false);
},
set: function(value){
return ns.setConfig("enablePopupScroll", value);
}
});
}(ns));
/*global window, define*/
/*jslint bitwise: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (ns) {
"use strict";
// Default configuration properties
ns.setConfig('rootDir', ns.getFrameworkPath(), true);
ns.setConfig('version', '');
ns.setConfig('allowCrossDomainPages', false, true);
ns.setConfig('domCache', false, true);
// .. other possible options
// ns.setConfig('autoBuildOnPageChange', true);
// ns.setConfig('autoInitializePage', true);
// ns.setConfig('container', document.body); // for defining application container
// ns.setConfig('pageContainer', document.body); // same as above, but for wearable version
}(ns));
/*global window, define*/
/*jslint bitwise: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* @class ns.support
* @author Maciej Urbanski <m.urbanski@samsung.com>
*/
(function (window, document, ns) {
"use strict";
var isTizen = !(typeof tizen === "undefined");
function isCircleShape() {
var testDiv = document.createElement("div"),
fakeBody = document.createElement("body"),
html = document.getElementsByTagName('html')[0],
style = getComputedStyle(testDiv),
isCircle;
testDiv.classList.add("is-circle-test");
fakeBody.appendChild(testDiv);
html.insertBefore(fakeBody, html.firstChild);
isCircle = style.width === "1px";
html.removeChild(fakeBody);
return isCircle;
}
ns.support = {
cssTransitions: true,
mediaquery: true,
cssPseudoElement: true,
touchOverflow: true,
cssTransform3d: true,
boxShadow: true,
scrollTop: 0,
dynamicBaseTag: true,
cssPointerEvents: false,
boundingRect: true,
browser: {
ie: false,
tizen: isTizen
},
shape: {
circle: isTizen ? window.matchMedia("(-tizen-geometric-shape: circle)").matches : isCircleShape(),
},
gradeA : function () {
return true;
}
};
}(window, window.document, ns));
/*global window, define*/
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint bitwise: true */
/*
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (ns) {
"use strict";
// Default configuration properties for wearable
ns.setConfig("autoBuildOnPageChange", false, true);
if(ns.support.shape.circle) {
ns.setConfig("pageTransition", "pop");
ns.setConfig("popupTransition", "pop");
ns.setConfig("popupFullSize", true);
ns.setConfig("scrollEndEffectArea", "screen");
ns.setConfig("enablePageScroll", true);
ns.setConfig("enablePopupScroll", true);
} else {
ns.setConfig("popupTransition", "slideup");
ns.setConfig("enablePageScroll", false);
ns.setConfig("enablePopupScroll", false);
}
// .. other possible options
// ns.setConfig('autoInitializePage', true);
// ns.setConfig('pageContainer', document.body); // defining application container for wearable
}(ns));
/*global window, define, XMLHttpRequest, console, Blob */
/*jslint nomen: true, browser: true, plusplus: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Utilities
*
* The Tizen Advanced UI (TAU) framework provides utilities for easy-developing
* and fully replaceable with jQuery method. When user using these DOM and
* selector methods, it provide more light logic and it proves performance
* of web app. The following table displays the utilities provided by the
* TAU framework.
*
* @class ns.util
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
*/
(function (window, document, ns) {
"use strict";
var currentFrame = null,
/**
* requestAnimationFrame function
* @method requestAnimationFrame
* @static
* @member ns.util
*/
requestAnimationFrame = (window.requestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.mozRequestAnimationFrame ||
window.oRequestAnimationFrame ||
window.msRequestAnimationFrame ||
function (callback) {
currentFrame = window.setTimeout(callback.bind(callback, +new Date()), 1000 / 60);
}).bind(window),
util = ns.util || {},
slice = [].slice;
/**
* fetchSync retrieves a text document synchronously, returns null on error
* @param {string} url
* @param {=string} [mime=""] Mime type of the resource
* @return {string|null}
* @static
* @member ns.util
*/
function fetchSync(url, mime) {
var xhr = new XMLHttpRequest(),
status;
xhr.open("get", url, false);
if (mime) {
xhr.overrideMimeType(mime);
}
xhr.send();
if (xhr.readyState === 4) {
status = xhr.status;
if (status === 200 || (status === 0 && xhr.responseText)) {
return xhr.responseText;
}
}
return null;
}
util.fetchSync = fetchSync;
/**
* Removes all script tags with src attribute from document and returns them
* @param {HTMLElement} container
* @return {Array.<HTMLElement>}
* @private
* @static
* @member ns.util
*/
function removeExternalScripts(container) {
var scripts = slice.call(container.querySelectorAll("script[src]")),
i = scripts.length,
script;
while (--i >= 0) {
script = scripts[i];
script.parentNode.removeChild(script);
}
return scripts;
}
/**
* Evaluates code, reason for a function is for an atomic call to evaluate code
* since most browsers fail to optimize functions with try-catch blocks, so this
* minimizes the effect, returns the function to run
* @param {string} code
* @return {Function}
* @static
* @member ns.util
*/
function safeEvalWrap(code) {
return function () {
try {
window.eval(code);
} catch (e) {
if (typeof console !== "undefined") {
if (e.stack) {
console.error(e.stack);
} else if (e.name && e.message) {
console.error(e.name, e.message);
} else {
console.error(e);
}
}
}
};
}
util.safeEvalWrap = safeEvalWrap;
/**
* Calls functions in supplied queue (array)
* @param {Array.<Function>} functionQueue
* @static
* @member ns.util
*/
function batchCall(functionQueue) {
var i,
length = functionQueue.length;
for (i = 0; i < length; ++i) {
functionQueue[i].call(window);
}
}
util.batchCall = batchCall;
/**
* Creates new script elements for scripts gathered from a differnt document
* instance, blocks asynchronous evaluation (by renaming src attribute) and
* returns an array of functions to run to evalate those scripts
* @param {Array.<HTMLElement>} scripts
* @param {HTMLElement} container
* @return {Array.<Function>}
* @private
* @static
* @member ns.util
*/
function createScriptsSync(scripts, container) {
var scriptElement,
scriptBody,
i,
length,
queue = [];
// proper order of execution
for (i = 0, length = scripts.length; i < length; ++i) {
scriptBody = fetchSync(scripts[i].src, "text/plain");
if (scriptBody) {
scriptElement = document.adoptNode(scripts[i]);
scriptElement.setAttribute("data-src", scripts[i].src);
scriptElement.removeAttribute("src"); // block evaluation
queue.push(safeEvalWrap(scriptBody));
if (container) {
container.appendChild(scriptElement);
}
}
}
return queue;
}
util.requestAnimationFrame = requestAnimationFrame;
/**
* cancelAnimationFrame function
* @method cancelAnimationFrame
* @return {Function}
* @member ns.util
* @static
*/
util.cancelAnimationFrame = (window.cancelAnimationFrame ||
window.webkitCancelAnimationFrame ||
window.mozCancelAnimationFrame ||
window.oCancelAnimationFrame ||
window.msCancelAnimationFrame ||
function () {
// propably wont work if there is any more than 1
// active animationFrame but we are trying anyway
window.clearTimeout(currentFrame);
}).bind(window);
/**
* Method make asynchronous call of function
* @method async
* @inheritdoc #requestAnimationFrame
* @member ns.util
* @static
*/
util.async = requestAnimationFrame;
/**
* Appends element from different document instance to current document in the
* container element and evaluates scripts (synchronously)
* @param {HTMLElement} element
* @param {HTMLElement} container
* @method importEvaluateAndAppendElement
* @member ns.util
* @static
*/
util.importEvaluateAndAppendElement = function (element, container) {
var externalScriptsQueue = createScriptsSync(removeExternalScripts(element), element),
newNode = document.importNode(element, true);
container.appendChild(newNode); // append and eval inline
batchCall(externalScriptsQueue);
return newNode;
};
/**
* Checks if specified string is a number or not
* @method isNumber
* @return {boolean}
* @member ns.util
* @static
*/
util.isNumber = function (query) {
var parsed = parseFloat(query);
return !isNaN(parsed) && isFinite(parsed);
};
/**
* Reappend script tags to DOM structure to correct run script
* @method runScript
* @param {string} baseUrl
* @param {HTMLScriptElement} script
* @member ns.util
* @deprecated 2.3
*/
util.runScript = function (baseUrl, script) {
var newScript = document.createElement("script"),
scriptData = null,
i,
scriptAttributes = slice.call(script.attributes),
src = script.getAttribute("src"),
path = util.path,
request,
attribute,
status;
// 'src' may become null when none src attribute is set
if (src !== null) {
src = path.makeUrlAbsolute(src, baseUrl);
}
//Copy script tag attributes
i = scriptAttributes.length;
while (--i >= 0) {
attribute = scriptAttributes[i];
if (attribute.name !== "src") {
newScript.setAttribute(attribute.name, attribute.value);
} else {
newScript.setAttribute("data-src", attribute.value);
}
}
if (src) {
scriptData = fetchSync(src, "text/plain");
} else {
scriptData = script.textContent;
}
if (scriptData) {
// add the returned content to a newly created script tag
newScript.src = URL.createObjectURL(new Blob([scriptData], {type: "text/javascript"}));
newScript.textContent = scriptData; // for compatibility with some libs ex. templating systems
}
script.parentNode.replaceChild(newScript, script);
};
ns.util = util;
}(window, window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Array Utility
* Utility helps work with arrays.
* @class ns.util.array
*/
(function (window, document, ns) {
"use strict";
/**
* Create an array containing the range of integers or characters
* from low to high (inclusive)
* @method range
* @param {number|string} low
* @param {number|string} high
* @param {number} step
* @static
* @return {Array} array containing continous elements
* @member ns.util.array
*/
function range(low, high, step) {
// Create an array containing the range of integers or characters
// from low to high (inclusive)
//
// version: 1107.2516
// discuss at: http://phpjs.org/functions/range
// + original by: Waldo Malqui Silva
// * example 1: range ( 0, 12 );
// * returns 1: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]
// * example 2: range( 0, 100, 10 );
// * returns 2: [0, 10, 20, 30, 40, 50, 60, 70, 80, 90, 100]
// * example 3: range( 'a', 'i' );
// * returns 3: ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i']
// * example 4: range( 'c', 'a' );
// * returns 4: ['c', 'b', 'a']
var matrix = [],
inival,
endval,
plus,
walker = step || 1,
chars = false;
if (!isNaN(low) && !isNaN(high)) {
inival = low;
endval = high;
} else if (isNaN(low) && isNaN(high)) {
chars = true;
inival = low.charCodeAt(0);
endval = high.charCodeAt(0);
} else {
inival = (isNaN(low) ? 0 : low);
endval = (isNaN(high) ? 0 : high);
}
plus = inival <= endval;
if (plus) {
while (inival <= endval) {
matrix.push((chars ? String.fromCharCode(inival) : inival));
inival += walker;
}
} else {
while (inival >= endval) {
matrix.push((chars ? String.fromCharCode(inival) : inival));
inival -= walker;
}
}
return matrix;
}
/**
* Check object is arraylike (arraylike include array and
* collection)
* @method isArrayLike
* @param {Object} object
* @return {boolean} Whether arraylike object or not
* @member ns.util.array
* @static
*/
function isArrayLike(object) {
var type = typeof object,
length = object && object.length;
// if object exists and is different from window
// window object has length property
if (object && object !== object.window) {
// If length value is not number, object is not array and collection.
// Collection type is not array but has length value.
// e.g) Array.isArray(document.childNodes) ==> false
return Array.isArray(object) || object instanceof NodeList || type === "function" &&
(length === 0 || typeof length === "number" && length > 0 && (length - 1) in object);
}
return false;
}
/**
* Faster version of standard forEach method in array
* Confirmed that this method is 20 times faster then native
* @method forEach
* @param {Array} array
* @param {Function} callback
* @member ns.util.array
* @static
*/
function forEach(array, callback) {
var i,
length;
if (!(array instanceof Array)) {
array = [].slice.call(array);
}
length = array.length;
for (i = 0; i < length; i++) {
callback(array[i], i, array);
}
}
/**
* Faster version of standard filter method in array
* @method filter
* @param {Array} array
* @param {Function} callback
* @member ns.util.array
* @static
*/
function filter(array, callback) {
var result = [],
i,
length,
value;
if (!(array instanceof Array)) {
array = [].slice.call(array);
}
length = array.length;
for (i = 0; i < length; i++) {
value = array[i];
if (callback(value, i, array)) {
result.push(value);
}
}
return result;
}
/**
* Faster version of standard map method in array
* Confirmed that this method is 60% faster then native
* @method map
* @param {Array} array
* @param {Function} callback
* @member ns.util.array
* @static
*/
function map(array, callback) {
var result = [],
i,
length;
if (!(array instanceof Array)) {
array = [].slice.call(array);
}
length = array.length;
for (i = 0; i < length; i++) {
result.push(callback(array[i], i, array));
}
return result;
}
/**
* Faster version of standard reduce method in array
* Confirmed that this method is 60% faster then native
* @method reduce
* @param {Array} array
* @param {Function} callback
* @param {*} [initialValue]
* @member ns.util.array
* @return {*}
* @static
*/
function reduce(array, callback, initialValue) {
var i,
length,
value,
result = initialValue;
if (!(array instanceof Array)) {
array = [].slice.call(array);
}
length = array.length;
for (i = 0; i < length; i++) {
value = array[i];
if (result === undefined && i === 0) {
result = value;
} else {
result = callback(result, value, i, array);
}
}
return result;
}
ns.util.array = {
range: range,
isArrayLike: isArrayLike,
forEach: forEach,
filter: filter,
map: map,
reduce: reduce
};
}(window, window.document, ns));
/*global window, ns, define, CustomEvent */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Events
*
* The Tizen Advanced UI (TAU) framework provides events optimized for the Tizen
* Web application. The following table displays the events provided by the TAU
* framework.
* @class ns.event
*/
(function (window, ns) {
"use strict";
/**
* Checks if specified variable is a array or not
* @method isArray
* @return {boolean}
* @member ns.event
* @private
* @static
*/
var isArray = Array.isArray,
isArrayLike = ns.util.array.isArrayLike,
/**
* @property {RegExp} SPLIT_BY_SPACES_REGEXP
*/
SPLIT_BY_SPACES_REGEXP = /\s+/g,
/**
* Returns trimmed value
* @method trim
* @param {string} value
* @return {string} trimmed string
* @static
* @private
* @member ns.event
*/
trim = function (value) {
return value.trim();
},
/**
* Split string to array
* @method getEventsListeners
* @param {string|Array|Object} names string with one name of event, many names of events divided by spaces, array with names of widgets or object in which keys are names of events and values are callbacks
* @param {Function} globalListener
* @return {Array}
* @static
* @private
* @member ns.event
*/
getEventsListeners = function (names, globalListener) {
var name,
result = [],
i;
if (typeof names === 'string') {
names = names.split(SPLIT_BY_SPACES_REGEXP).map(trim);
}
if (isArray(names)) {
for (i=0; i<names.length; i++) {
result.push({type: names[i], callback: globalListener});
}
} else {
for (name in names) {
if (names.hasOwnProperty(name)) {
result.push({type: name, callback: names[name]});
}
}
}
return result;
};
ns.event = {
/**
* Triggers custom event fastOn element
* The return value is false, if at least one of the event
* handlers which handled this event, called preventDefault.
* Otherwise it returns true.
* @method trigger
* @param {HTMLElement} element
* @param {string} type
* @param {?*} [data=null]
* @param {boolean=} [bubbles=true]
* @param {boolean=} [cancelable=true]
* @return {boolean=}
* @member ns.event
* @static
*/
trigger: function (element, type, data, bubbles, cancelable) {
var evt = new CustomEvent(type, {
"detail": data,
//allow event to bubble up, required if we want to allow to listen fastOn document etc
bubbles: typeof bubbles === "boolean" ? bubbles : true,
cancelable: typeof cancelable === "boolean" ? cancelable : true
});
return element.dispatchEvent(evt);
},
/**
* Prevent default on original event
* @method preventDefault
* @param {CustomEvent} event
* @member ns.event
* @static
*/
preventDefault: function (event) {
var originalEvent = event._originalEvent;
// @todo this.isPropagationStopped = returnTrue;
if (originalEvent && originalEvent.preventDefault) {
originalEvent.preventDefault();
}
event.preventDefault();
},
/**
* Stop event propagation
* @method stopPropagation
* @param {CustomEvent} event
* @member ns.event
* @static
*/
stopPropagation: function (event) {
var originalEvent = event._originalEvent;
// @todo this.isPropagationStopped = returnTrue;
if (originalEvent && originalEvent.stopPropagation) {
originalEvent.stopPropagation();
}
event.stopPropagation();
},
/**
* Stop event propagation immediately
* @method stopImmediatePropagation
* @param {CustomEvent} event
* @member ns.event
* @static
*/
stopImmediatePropagation: function (event) {
var originalEvent = event._originalEvent;
// @todo this.isPropagationStopped = returnTrue;
if (originalEvent && originalEvent.stopImmediatePropagation) {
originalEvent.stopImmediatePropagation();
}
event.stopImmediatePropagation();
},
/**
* Return document relative cords for event
* @method documentRelativeCoordsFromEvent
* @param {Event} event
* @return {Object}
* @return {number} return.x
* @return {number} return.y
* @member ns.event
* @static
*/
documentRelativeCoordsFromEvent: function(event) {
var _event = event ? event : window.event,
client = {
x: _event.clientX,
y: _event.clientY
},
page = {
x: _event.pageX,
y: _event.pageY
},
posX = 0,
posY = 0,
touch0,
body = document.body,
documentElement = document.documentElement;
if (event.type.match(/^touch/)) {
touch0 = _event.targetTouches[0] || _event.originalEvent.targetTouches[0];
page = {
x: touch0.pageX,
y: touch0.pageY
};
client = {
x: touch0.clientX,
y: touch0.clientY
};
}
if (page.x || page.y) {
posX = page.x;
posY = page.y;
}
else if (client.x || client.y) {
posX = client.x + body.scrollLeft + documentElement.scrollLeft;
posY = client.y + body.scrollTop + documentElement.scrollTop;
}
return { x: posX, y: posY };
},
/**
* Return target relative cords for event
* @method targetRelativeCoordsFromEvent
* @param {Event} event
* @return {Object}
* @return {number} return.x
* @return {number} return.y
* @member ns.event
* @static
*/
targetRelativeCoordsFromEvent: function(event) {
var target = event.target,
cords = {
x: event.offsetX,
y: event.offsetY
};
if (cords.x === undefined || isNaN(cords.x) ||
cords.y === undefined || isNaN(cords.y)) {
cords = ns.event.documentRelativeCoordsFromEvent(event);
cords.x -= target.offsetLeft;
cords.y -= target.offsetTop;
}
return cords;
},
/**
* Add event listener to element
* @method fastOn
* @param {HTMLElement} element
* @param {string} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
fastOn: function(element, type, listener, useCapture) {
element.addEventListener(type, listener, useCapture || false);
},
/**
* Remove event listener to element
* @method fastOff
* @param {HTMLElement} element
* @param {string} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
fastOff: function(element, type, listener, useCapture) {
element.removeEventListener(type, listener, useCapture || false);
},
/**
* Add event listener to element with prefixes for all browsers
* @method fastPrefixedOn
* @param {HTMLElement} element
* @param {string} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
prefixedFastOn: function(element, type, listener, useCapture) {
var nameForPrefix = type.charAt(0).toLocaleUpperCase() + type.substring(1);
element.addEventListener(type.toLowerCase(), listener, useCapture || false);
element.addEventListener("webkit" + nameForPrefix, listener, useCapture || false);
element.addEventListener("moz" + nameForPrefix, listener, useCapture || false);
element.addEventListener("ms" + nameForPrefix, listener, useCapture || false);
element.addEventListener("o" + nameForPrefix.toLowerCase(), listener, useCapture || false);
},
/**
* Remove event listener to element with prefixes for all browsers
* @method fastPrefixedOff
* @param {HTMLElement} element
* @param {string} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
prefixedFastOff: function(element, type, listener, useCapture) {
var nameForPrefix = type.charAt(0).toLocaleUpperCase() + type.substring(1);
element.removeEventListener(type.toLowerCase(), listener, useCapture || false);
element.removeEventListener("webkit" + nameForPrefix, listener, useCapture || false);
element.removeEventListener("moz" + nameForPrefix, listener, useCapture || false);
element.removeEventListener("ms" + nameForPrefix, listener, useCapture || false);
element.removeEventListener("o" + nameForPrefix.toLowerCase(), listener, useCapture || false);
},
/**
* Add event listener to element that can be added addEventListner
* @method on
* @param {HTMLElement|HTMLDocument|Window} element
* @param {string|Array|Object} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
on: function(element, type, listener, useCapture) {
var i,
j,
elementsLength,
typesLength,
elements,
listeners;
if (isArrayLike(element)) {
elements = element;
} else {
elements = [element];
}
elementsLength = elements.length;
listeners = getEventsListeners(type, listener);
typesLength = listeners.length;
for (i = 0; i < elementsLength; i++) {
if (typeof elements[i].addEventListener === "function") {
for (j = 0; j < typesLength; j++) {
ns.event.fastOn(elements[i], listeners[j].type, listeners[j].callback, useCapture);
}
}
}
},
/**
* Remove event listener to element
* @method off
* @param {HTMLElement|HTMLDocument|Window} element
* @param {string|Array|Object} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
off: function(element, type, listener, useCapture) {
var i,
j,
elementsLength,
typesLength,
elements,
listeners;
if (isArrayLike(element)) {
elements = element;
} else {
elements = [element];
}
elementsLength = elements.length;
listeners = getEventsListeners(type, listener);
typesLength = listeners.length;
for (i = 0; i < elementsLength; i++) {
if (typeof elements[i].addEventListener === "function") {
for (j = 0; j < typesLength; j++) {
ns.event.fastOff(elements[i], listeners[j].type, listeners[j].callback, useCapture);
}
}
}
},
/**
* Add event listener to element only for one trigger
* @method one
* @param {HTMLElement|HTMLDocument|window} element
* @param {string|Array|Object} type
* @param {Function} listener
* @param {boolean} [useCapture=false]
* @member ns.event
* @static
*/
one: function(element, type, listener, useCapture) {
var arraySlice = [].slice,
i,
j,
elementsLength,
typesLength,
elements,
types,
listeners,
callbacks = [];
if (isArrayLike(element)) {
elements = arraySlice.call(element);
} else {
elements = [element];
}
elementsLength = elements.length;
listeners = getEventsListeners(type, listener);
typesLength = listeners.length;
for (i = 0; i < elementsLength; i++) {
if (typeof elements[i].addEventListener === "function") {
callbacks[i] = [];
for (j = 0; j < typesLength; j++) {
callbacks[i][j] = (function(i, j) {
var args = arraySlice.call(arguments);
ns.event.fastOff(elements[i], listeners[j].type, callbacks[i][j], useCapture);
args.shift(); // remove the first argument of binding function
args.shift(); // remove the second argument of binding function
listeners[j].callback.apply(this, args);
}).bind(null, i, j);
ns.event.fastOn(elements[i], listeners[j].type, callbacks[i][j], useCapture);
}
}
}
}
};
}(window, ns));
/*global window, ns, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Info
*
* Various TAU information
* @class ns.info
*/
(function (window, document, ns) {
"use strict";
/**
* @property {Object} info
* @property {string} [info.profile="default"] Current runtime profile
* @property {string} [info.theme="default"] Current runtime theme
* @property {string} info.version Current runtime version
* @member ns.info
* @static
*/
var eventUtils = ns.event,
info = {
profile: "default",
theme: "default",
version: ns.version,
/**
* Refreshes information about runtime
* @method refreshTheme
* @param {Function} done Callback run when the theme is discovered
* @member ns.info
* @return {null|String}
* @static
*/
refreshTheme: function (done) {
var el = document.createElement("span"),
parent = document.body,
themeName = null;
if (document.readyState !== "interactive" && document.readyState !== "complete") {
eventUtils.fastOn(document, "DOMContentLoaded", this.refreshTheme.bind(this, done));
return null;
}
el.classList.add("tau-info-theme");
parent.appendChild(el);
themeName = window.getComputedStyle(el, ":after").content;
parent.removeChild(el);
if (themeName && themeName.length > 0) {
this.theme = themeName;
}
themeName = themeName || null;
if (done) {
done(themeName);
}
return themeName;
}
};
info.refreshTheme();
ns.info = info;
}(window, window.document, ns));
/*global define: true, window: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Selectors Utility
* Object contains functions to get HTML elements by different selectors.
* @class ns.util.selectors
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Jadwiga Sosnowska <j.sosnowska@partner.samsung.com>
* @author Damian Osipiuk <d.osipiuk@samsung.com>
*/
(function (document, ns) {
"use strict";
/**
* @method slice Alias for array slice method
* @member ns.util.selectors
* @private
* @static
*/
var slice = [].slice,
/**
* @method matchesSelectorType
* @return {string|boolean}
* @member ns.util.selectors
* @private
* @static
*/
matchesSelectorType = (function () {
var el = document.createElement("div");
if (typeof el.webkitMatchesSelector === "function") {
return "webkitMatchesSelector";
}
if (typeof el.mozMatchesSelector === "function") {
return "mozMatchesSelector";
}
if (typeof el.msMatchesSelector === "function") {
return "msMatchesSelector";
}
if (typeof el.matchesSelector === "function") {
return "matchesSelector";
}
if (typeof el.matches === "function") {
return "matches";
}
return false;
}());
/**
* Prefix selector with 'data-' and namespace if present
* @method getDataSelector
* @param {string} selector
* @return {string}
* @member ns.util.selectors
* @private
* @static
*/
function getDataSelector(selector) {
var namespace = ns.getConfig('namespace');
return '[data-' + (namespace ? namespace + '-' : '') + selector + ']';
}
/**
* Runs matches implementation of matchesSelector
* method on specified element
* @method matchesSelector
* @param {HTMLElement} element
* @param {string} selector
* @return {boolean}
* @static
* @member ns.util.selectors
*/
function matchesSelector(element, selector) {
if (matchesSelectorType && element[matchesSelectorType]) {
return element[matchesSelectorType](selector);
}
return false;
}
/**
* Return array with all parents of element.
* @method parents
* @param {HTMLElement} element
* @return {Array}
* @member ns.util.selectors
* @private
* @static
*/
function parents(element) {
var items = [],
current = element.parentNode;
while (current && current !== document) {
items.push(current);
current = current.parentNode;
}
return items;
}
/**
* Checks if given element and its ancestors matches given function
* @method closest
* @param {HTMLElement} element
* @param {Function} testFunction
* @return {?HTMLElement}
* @member ns.util.selectors
* @static
* @private
*/
function closest(element, testFunction) {
var current = element;
while (current && current !== document) {
if (testFunction(current)) {
return current;
}
current = current.parentNode;
}
return null;
}
/**
* @method testSelector
* @param {string} selector
* @param {HTMLElement} node
* @return {boolean}
* @member ns.util.selectors
* @static
* @private
*/
function testSelector(selector, node) {
return matchesSelector(node, selector);
}
/**
* @method testClass
* @param {string} className
* @param {HTMLElement} node
* @return {boolean}
* @member ns.util.selectors
* @static
* @private
*/
function testClass(className, node) {
return node && node.classList && node.classList.contains(className);
}
/**
* @method testTag
* @param {string} tagName
* @param {HTMLElement} node
* @return {boolean}
* @member ns.util.selectors
* @static
* @private
*/
function testTag(tagName, node) {
return node.tagName.toLowerCase() === tagName;
}
/**
* @class ns.util.selectors
*/
ns.util.selectors = {
matchesSelector: matchesSelector,
/**
* Return array with children pass by given selector.
* @method getChildrenBySelector
* @param {HTMLElement} context
* @param {string} selector
* @return {Array}
* @static
* @member ns.util.selectors
*/
getChildrenBySelector: function (context, selector) {
return slice.call(context.children).filter(testSelector.bind(null, selector));
},
/**
* Return array with children pass by given data-namespace-selector.
* @method getChildrenByDataNS
* @param {HTMLElement} context
* @param {string} dataSelector
* @return {Array}
* @static
* @member ns.util.selectors
*/
getChildrenByDataNS: function (context, dataSelector) {
return slice.call(context.children).filter(testSelector.bind(null, getDataSelector(dataSelector)));
},
/**
* Return array with children with given class name.
* @method getChildrenByClass
* @param {HTMLElement} context
* @param {string} className
* @return {Array}
* @static
* @member ns.util.selectors
*/
getChildrenByClass: function (context, className) {
return slice.call(context.children).filter(testClass.bind(null, className));
},
/**
* Return array with children with given tag name.
* @method getChildrenByTag
* @param {HTMLElement} context
* @param {string} tagName
* @return {Array}
* @static
* @member ns.util.selectors
*/
getChildrenByTag: function (context, tagName) {
return slice.call(context.children).filter(testTag.bind(null, tagName));
},
/**
* Return array with all parents of element.
* @method getParents
* @param {HTMLElement} context
* @param {string} selector
* @return {Array}
* @static
* @member ns.util.selectors
*/
getParents: parents,
/**
* Return array with all parents of element pass by given selector.
* @method getParentsBySelector
* @param {HTMLElement} context
* @param {string} selector
* @return {Array}
* @static
* @member ns.util.selectors
*/
getParentsBySelector: function (context, selector) {
return parents(context).filter(testSelector.bind(null, selector));
},
/**
* Return array with all parents of element pass by given selector with namespace.
* @method getParentsBySelectorNS
* @param {HTMLElement} context
* @param {string} selector
* @return {Array}
* @static
* @member ns.util.selectors
*/
getParentsBySelectorNS: function (context, selector) {
return parents(context).filter(testSelector.bind(null, getDataSelector(selector)));
},
/**
* Return array with all parents of element with given class name.
* @method getParentsByClass
* @param {HTMLElement} context
* @param {string} className
* @return {Array}
* @static
* @member ns.util.selectors
*/
getParentsByClass: function (context, className) {
return parents(context).filter(testClass.bind(null, className));
},
/**
* Return array with all parents of element with given tag name.
* @method getParentsByTag
* @param {HTMLElement} context
* @param {string} tagName
* @return {Array}
* @static
* @member ns.util.selectors
*/
getParentsByTag: function (context, tagName) {
return parents(context).filter(testTag.bind(null, tagName));
},
/**
* Return first element from parents of element pass by selector.
* @method getClosestBySelector
* @param {HTMLElement} context
* @param {string} selector
* @return {HTMLElement}
* @static
* @member ns.util.selectors
*/
getClosestBySelector: function (context, selector) {
return closest(context, testSelector.bind(null, selector));
},
/**
* Return first element from parents of element pass by selector with namespace.
* @method getClosestBySelectorNS
* @param {HTMLElement} context
* @param {string} selector
* @return {HTMLElement}
* @static
* @member ns.util.selectors
*/
getClosestBySelectorNS: function (context, selector) {
return closest(context, testSelector.bind(null, getDataSelector(selector)));
},
/**
* Return first element from parents of element with given class name.
* @method getClosestByClass
* @param {HTMLElement} context
* @param {string} selector
* @return {HTMLElement}
* @static
* @member ns.util.selectors
*/
getClosestByClass: function (context, selector) {
return closest(context, testClass.bind(null, selector));
},
/**
* Return first element from parents of element with given tag name.
* @method getClosestByTag
* @param {HTMLElement} context
* @param {string} selector
* @return {HTMLElement}
* @static
* @member ns.util.selectors
*/
getClosestByTag: function (context, selector) {
return closest(context, testTag.bind(null, selector));
},
/**
* Return array of elements from context with given data-selector
* @method getAllByDataNS
* @param {HTMLElement} context
* @param {string} dataSelector
* @return {Array}
* @static
* @member ns.util.selectors
*/
getAllByDataNS: function (context, dataSelector) {
return slice.call(context.querySelectorAll(getDataSelector(dataSelector)));
},
/**
* Get scrollable parent elmenent
* @method getScrollableParent
* @param {HTMLElement} element
* @return {HTMLElement}
* @static
* @member ns.util.selectors
*/
getScrollableParent: function (element) {
var overflow,
style;
while (element && element != document.body) {
style = window.getComputedStyle(element);
if (style) {
overflow = style.getPropertyValue("overflow-y");
if (overflow === "scroll" || (overflow === "auto" && element.scrollHeight > element.clientHeight)) {
return element;
}
}
element = element.parentNode;
}
return null;
}
};
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Object Utility
* Object contains functions help work with objects.
* @class ns.util.object
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (ns) {
"use strict";
var object = {
/**
* Copy object to new object
* @method copy
* @param {Object} orgObject
* @return {Object}
* @static
* @member ns.util.object
*/
copy: function (orgObject) {
return object.merge({}, orgObject);
},
/**
* Attach fields from second object to first object.
* @method fastMerge
* @param {Object} newObject
* @param {Object} orgObject
* @return {Object}
* @static
* @member ns.util.object
*/
fastMerge: function (newObject, orgObject) {
var key;
for (key in orgObject) {
if (orgObject.hasOwnProperty(key)) {
newObject[key] = orgObject[key];
}
}
return newObject;
},
/**
* Attach fields from second and next object to first object.
* @method merge
* @param {Object} newObject
* @param {...Object} orgObject
* @param {?boolean} [override=true]
* @return {Object}
* @static
* @member ns.util.object
*/
merge: function ( /* newObject, orgObject, override */ ) {
var newObject, orgObject, override,
key,
args = [].slice.call(arguments),
argsLength = args.length,
i;
newObject = args.shift();
override = true;
if (typeof arguments[argsLength-1] === "boolean") {
override = arguments[argsLength-1];
argsLength--;
}
for (i = 0; i < argsLength; i++) {
orgObject = args.shift();
if (orgObject !== null) {
for (key in orgObject) {
if (orgObject.hasOwnProperty(key) && ( override || newObject[key] === undefined )) {
newObject[key] = orgObject[key];
}
}
}
}
return newObject;
},
/**
* Function add to Constructor prototype Base object and add to prototype properties and methods from
* prototype object.
* @method inherit
* @param {Function} Constructor
* @param {Function} Base
* @param {Object} prototype
* @static
* @member ns.util.object
*/
/* jshint -W083 */
inherit: function( Constructor, Base, prototype ) {
var basePrototype = new Base(),
property,
value;
for (property in prototype) {
if (prototype.hasOwnProperty(property)) {
value = prototype[property];
if ( typeof value === "function" ) {
basePrototype[property] = (function createFunctionWithSuper(Base, property, value) {
var _super = function() {
var superFunction = Base.prototype[property];
if (superFunction) {
return superFunction.apply(this, arguments);
}
return null;
};
return function() {
var __super = this._super,
returnValue;
this._super = _super;
returnValue = value.apply(this, arguments);
this._super = __super;
return returnValue;
};
}(Base, property, value));
} else {
basePrototype[property] = value;
}
}
}
Constructor.prototype = basePrototype;
Constructor.prototype.constructor = Constructor;
},
/**
* Returns true if every property value corresponds value from 'value' argument
* @method hasPropertiesOfValue
* @param {Object} obj
* @param {*} [value=undefined]
* @return {boolean}
*/
hasPropertiesOfValue: function (obj, value) {
var keys = Object.keys(obj),
i = keys.length;
// Empty array should return false
if (i === 0) {
return false;
}
while (--i >= 0) {
if (obj[keys[i]] !== value) {
return false;
}
}
return true;
},
/**
* Remove properties from object.
* @method removeProperties
* @param {Object} object
* @param {Array} propertiesToRemove
* @return {Object}
*/
removeProperties: function (object, propertiesToRemove) {
var length = propertiesToRemove.length,
property,
i;
for (i = 0; i < length; i++) {
property = propertiesToRemove[i];
if (object.hasOwnProperty(property)) {
delete object[property];
}
}
return object;
}
};
ns.util.object = object;
}(ns));
/*global window, define, ns, Node, HTMLElement */
/*jslint nomen: true, plusplus: true, bitwise: false */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Engine
* Main class with engine of library which control communication
* between parts of framework.
* @class ns.engine
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Michal Szepielak <m.szepielak@samsung.com>
* @author Jadwiga Sosnowska <j.sosnowska@partner.samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
* @author Tomasz Lukawski <t.lukawski@samsung.com>
* @author Przemyslaw Ciezkowski <p.ciezkowski@samsung.com>
*/
(function (window, document, ns) {
"use strict";
/**
* @method slice Array.slice
* @private
* @static
* @member ns.engine
*/
var slice = [].slice,
/**
* @property {Object} eventUtils {@link ns.event}
* @private
* @static
* @member ns.engine
*/
eventUtils = ns.event,
objectUtils = ns.util.object,
selectors = ns.util.selectors,
/**
* @property {Object} widgetDefs Object with widgets definitions
* @private
* @static
* @member ns.engine
*/
widgetDefs = {},
/**
* @property {Object} widgetBindingMap Object with widgets bindings
* @private
* @static
* @member ns.engine
*/
widgetBindingMap = {},
location = window.location,
/**
* engine mode, if true then engine only builds widgets
* @property {boolean} justBuild
* @private
* @static
* @member ns.engine
*/
justBuild = location.hash === "#build",
/**
* @property {string} [TYPE_STRING="string"] local cache of string type name
* @private
* @static
* @readonly
* @member ns.engine
*/
TYPE_STRING = "string",
/**
* @property {string} [TYPE_FUNCTION="function"] local cache of function type name
* @private
* @static
* @readonly
* @member ns.engine
*/
TYPE_FUNCTION = "function",
/**
* @property {string} [DATA_BUILT="data-tau-built"] attribute informs that widget id build
* @private
* @static
* @readonly
* @member ns.engine
*/
DATA_BUILT = "data-tau-built",
/**
* @property {string} [DATA_NAME="data-tau-name"] attribute contains widget name
* @private
* @static
* @readonly
* @member ns.engine
*/
DATA_NAME = "data-tau-name",
/**
* @property {string} [DATA_BOUND="data-tau-bound"] attribute informs that widget id bound
* @private
* @static
* @readonly
* @member ns.engine
*/
DATA_BOUND = "data-tau-bound",
/**
* @property {string} NAMES_SEPARATOR
* @private
* @static
* @readonly
*/
NAMES_SEPARATOR = ",",
/**
* @property {string} [querySelectorWidgets="*[data-tau-built][data-tau-name]:not([data-tau-bound])"] query selector for all widgets which are built but not bound
* @private
* @static
* @member ns.engine
*/
// @TODO this selector is not valid ...
querySelectorWidgets = "*[" + DATA_BUILT + "][" + DATA_NAME + "]:not([" + DATA_BOUND + "])",
/**
* @method excludeBuildAndBound
* @private
* @static
* @member ns.engine
* @return {string} :not([data-tau-built*='widgetName']):not([data-tau-bound*='widgetName'])
*/
excludeBuiltAndBound = function (widgetType) {
return ":not([" + DATA_BUILT + "*='" + widgetType +"']):not([" + DATA_BOUND + "*='" + widgetType +"'])";
},
/**
* Engine event types
* @property {Object} eventType
* @property {string} eventType.INIT="tauinit" INIT of framework init event
* @property {string} eventType.WIDGET_BOUND="widgetbound" WIDGET_BOUND of widget bound event
* @property {string} eventType.WIDGET_DEFINED="widgetdefined" WIDGET_DEFINED of widget built event
* @property {string} eventType.WIDGET_BUILT="widgetbuilt" WIDGET_BUILT of widget built event
* @property {string} eventType.BOUND="bound" BOUND of bound event
* @static
* @readonly
* @member ns.engine
*/
eventType = {
INIT: "tauinit",
WIDGET_BOUND: "widgetbound",
WIDGET_DEFINED: "widgetdefined",
WIDGET_BUILT: "widgetbuilt",
BOUND: "bound"
},
engine,
/**
* @property {Object} router Router object
* @private
* @static
* @member ns.engine
*/
router;
/**
* This function prepares selector for widget' definition
* @method selectorChange
* @param {string} selectorName
* @return {string} new selector
* @member ns.engine
* @static
*/
function selectorChange (selectorName) {
if (selectorName.match(/\[data-role=/) && !selectorName.match(/:not\(\[data-role=/)) {
return selectorName.trim();
}
return selectorName.trim() + ":not([data-role='none'])";
}
/**
* Function to define widget
* @method defineWidget
* @param {string} name
* @param {string} selector
* @param {Array} methods
* @param {Object} widgetClass
* @param {string} [namespace]
* @param {boolean} [redefine]
* @param {boolean} [widgetNameToLowercase = true]
* @return {boolean}
* @member ns.engine
* @static
*/
function defineWidget(name, selector, methods, widgetClass, namespace, redefine, widgetNameToLowercase) {
var definition;
// Widget name is absolutely required
if (name) {
if (!widgetDefs[name] || redefine) {
methods = methods || [];
methods.push("destroy", "disable", "enable", "option", "refresh", "value");
definition = {
name: name,
methods: methods,
selector: selector || "",
selectors: selector ? selector.split(",").map(selectorChange) : [],
widgetClass: widgetClass || null,
namespace: namespace || "",
widgetNameToLowercase: widgetNameToLowercase === undefined ? true : !!widgetNameToLowercase
};
widgetDefs[name] = definition;
eventUtils.trigger(document, "widgetdefined", definition, false);
return true;
}
} else {
ns.error("Widget with selector [" + selector + "] defined without a name, aborting!");
}
return false;
}
/**
* Get binding for element
* @method getBinding
* @static
* @param {HTMLElement|string} element
* @param {string} [type] widget name
* @return {?Object}
* @member ns.engine
*/
function getBinding(element, type) {
var id = !element || typeof element === TYPE_STRING ? element : element.id,
binding,
widgetInstance,
bindingElement,
storedWidgetNames;
if (typeof element === TYPE_STRING) {
element = document.getElementById(id);
}
// Fetch group of widget defined for this element
binding = widgetBindingMap[id];
if (binding && typeof binding === "object") {
// If name is defined it's possible to fetch it instantly
if (type) {
widgetInstance = binding.instances[type];
} else {
storedWidgetNames = Object.keys(binding.instances);
widgetInstance = binding.instances[storedWidgetNames[0]];
}
// Return only it instance of the proper widget exists
if (widgetInstance) {
// Check if widget instance has that same object referenced
if (widgetInstance.element === element) {
return widgetInstance;
}
}
}
return null;
}
/**
* Set binding of widget
* @method setBinding
* @param {ns.widget.BaseWidget} widgetInstance
* @static
* @member ns.engine
*/
function setBinding(widgetInstance) {
var id = widgetInstance.element.id,
type = widgetInstance.name,
widgetBinding = widgetBindingMap[id];
// If the HTMLElement never had a widget declared create an empty object
if(!widgetBinding) {
widgetBinding = {
elementId: id,
element: widgetInstance.element,
instances: {}
};
}
widgetBinding.instances[type] = widgetInstance;
widgetBindingMap[id] = widgetBinding;
}
/**
* Returns all bindings for element or id gives as parameter
* @method getAllBindings
* @param {HTMLElement|string} element
* @return {?Object}
* @static
* @member ns.engine
*/
function getAllBindings(element) {
var id = !element || typeof element === TYPE_STRING ? element : element.id;
return (widgetBindingMap[id] && widgetBindingMap[id].instances) || null;
}
/**
* Removes given name from attributeValue string.
* Names should be separated with a NAMES_SEPARATOR
* @param {string} name
* @param {string} attributeValue
* @private
* @static
* @return {string}
*/
function _removeWidgetNameFromAttribute(name, attributeValue) {
var widgetNames,
searchResultIndex;
// Split attribute value by separator
widgetNames = attributeValue.split(NAMES_SEPARATOR);
searchResultIndex = widgetNames.indexOf(name);
if (searchResultIndex > -1) {
widgetNames.splice(searchResultIndex, 1);
attributeValue = widgetNames.join(NAMES_SEPARATOR);
}
return attributeValue;
}
function _removeAllBindingAttributes(element) {
element.removeAttribute(DATA_BUILT);
element.removeAttribute(DATA_BOUND);
element.removeAttribute(DATA_NAME);
}
/**
* Remove binding data attributes for element.
* @method _removeBindingAttributes
* @param {HTMLElement} element
* @param {string} type widget type (name)
* @private
* @static
* @member ns.engine
*/
function _removeWidgetFromAttributes(element, type) {
var dataBuilt,
dataBound,
dataName;
// Most often case is that name is not defined
if (!type) {
_removeAllBindingAttributes(element);
} else {
dataBuilt = _removeWidgetNameFromAttribute(type, element.getAttribute(DATA_BUILT) || "");
dataBound = _removeWidgetNameFromAttribute(type, element.getAttribute(DATA_BOUND) || "");
dataName = _removeWidgetNameFromAttribute(type, element.getAttribute(DATA_NAME) || "");
// Check if all attributes have at least one widget
if (dataBuilt && dataBound && dataName) {
element.setAttribute(DATA_BUILT, dataBuilt);
element.setAttribute(DATA_BOUND, dataBound);
element.setAttribute(DATA_NAME, dataName);
} else {
// If something is missing remove everything
_removeAllBindingAttributes(element);
}
}
}
/**
* Method removes binding for single widget.
* @method _removeSingleBinding
* @param {Object} bindingGroup
* @param {string} type
* @return {boolean}
* @private
* @static
*/
function _removeSingleBinding(bindingGroup, type) {
var widgetInstance = bindingGroup[type];
if (widgetInstance){
if (widgetInstance.element && typeof widgetInstance.element.setAttribute === TYPE_FUNCTION) {
_removeWidgetFromAttributes(widgetInstance.element, type);
}
bindingGroup[type] = null;
return true;
}
return false;
}
/**
* Remove binding for widget based on element.
* @method removeBinding
* @param {HTMLElement|string} element
* @param {string} type widget name
* @return {boolean}
* @static
* @member ns.engine
*/
function removeBinding(element, type) {
var id = (typeof element === TYPE_STRING) ? element : element.id,
binding = widgetBindingMap[id],
bindingGroup,
widgetName,
partialSuccess,
fullSuccess = false;
// [NOTICE] Due to backward compatibility calling removeBinding
// with one parameter should remove all bindings
if (binding) {
if (typeof element === TYPE_STRING) {
// Search based on current document may return bad results,
// use previously defined element if it exists
element = binding.element;
}
if (element) {
_removeWidgetFromAttributes(element, type);
}
bindingGroup = widgetBindingMap[id] && widgetBindingMap[id].instances;
if (bindingGroup) {
if (!type) {
fullSuccess = true;
// Iterate over group of created widgets
for (widgetName in bindingGroup) {
if (bindingGroup.hasOwnProperty(widgetName)) {
partialSuccess = _removeSingleBinding(bindingGroup, widgetName);
// As we iterate over keys we are sure we want to remove this element
// NOTE: Removing property by delete is slower than assigning null value
bindingGroup[widgetName] = null;
fullSuccess = (fullSuccess && partialSuccess);
}
}
// If the object bindingGroup is empty or every key has a null value
if (objectUtils.hasPropertiesOfValue(bindingGroup, null)) {
// NOTE: Removing property by delete is slower than assigning null value
widgetBindingMap[id] = null;
}
return fullSuccess;
}
partialSuccess = _removeSingleBinding(bindingGroup, type);
if (objectUtils.hasPropertiesOfValue(bindingGroup, null)) {
widgetBindingMap[id] = null;
}
return partialSuccess;
}
}
return false;
}
/**
* Removes all bindings of widgets.
* @method removeAllBindings
* @param {HTMLElement|string} element
* @return {boolean}
* @static
* @member ns.engine
*/
function removeAllBindings(element) {
// @TODO this should be coded in the other way around, removeAll should loop through all bindings and inside call removeBinding
// but due to backward compatibility that code should be more readable
return removeBinding(element);
}
/**
* If element not exist create base element for widget.
* @method ensureElement
* @param {HTMLElement} element
* @param {ns.widget.BaseWidget} Widget
* @return {HTMLElement}
* @static
* @private
* @member ns.engine
*/
function ensureElement(element, Widget) {
if (!element || !element instanceof HTMLElement) {
if (typeof Widget.createEmptyElement === TYPE_FUNCTION) {
element = Widget.createEmptyElement();
} else {
element = document.createElement("div");
}
}
return element;
}
/**
* Load widget
* @method processWidget
* @param {HTMLElement} element base element of widget
* @param {Object} definition definition of widget
* @param {ns.widget.BaseWidget} definition.widgetClass
* @param {string} definition.name
* @param {Object} [options] options for widget
* @private
* @static
* @member ns.engine
*/
function processWidget(element, definition, options) {
var widgetOptions = options || {},
createFunction = widgetOptions.create,
Widget = definition.widgetClass,
/**
* @type {ns.widget.BaseWidget} widgetInstance
*/
widgetInstance,
buildAttribute,
parentEnhance,
existingBinding;
element = ensureElement(element, Widget);
widgetInstance = Widget ? new Widget(element) : false;
// if any parent has attribute data-enhance=false then stop building widgets
parentEnhance = selectors.getParentsBySelectorNS(element, 'enhance=false');
// While processing widgets queue other widget may built this one before
// it reaches it's turn
existingBinding = getBinding(element, definition.name);
if (existingBinding && existingBinding.element === element) {
return existingBinding.element;
}
if (widgetInstance && !parentEnhance.length) {
widgetInstance.configure(definition, element, options);
// Run .create method from widget options when a [widgetName]create event is triggered
if (typeof createFunction === TYPE_FUNCTION) {
eventUtils.one(element, definition.name.toLowerCase() + "create", createFunction);
}
if (element.id) {
widgetInstance.id = element.id;
}
// Check if this type of widget was build for this element before
buildAttribute = element.getAttribute(DATA_BUILT);
if (!buildAttribute || (buildAttribute && buildAttribute.split(NAMES_SEPARATOR).indexOf(widgetInstance.name) === -1)) {
element = widgetInstance.build(element);
}
if (element) {
widgetInstance.element = element;
setBinding(widgetInstance);
widgetInstance.trigger(eventType.WIDGET_BUILT, widgetInstance, false);
if (!justBuild) {
widgetInstance.init(element);
}
widgetInstance.bindEvents(element, justBuild);
eventUtils.trigger(element, eventType.WIDGET_BOUND, widgetInstance, false);
eventUtils.trigger(document, eventType.WIDGET_BOUND, widgetInstance);
} else {
}
}
return widgetInstance.element;
}
/**
* Destroys widget of given 'type' for given HTMLElement.
* [NOTICE] This method won't destroy any children widgets.
* @method destroyWidget
* @param {HTMLElement|string} element
* @param {string} type
* @static
* @member ns.engine
*/
function destroyWidget(element, type) {
var widgetInstance;
if (typeof element === TYPE_STRING) {
element = document.getElementById(element);
}
// If type is not defined all widgets should be removed
// this is for backward compatibility
widgetInstance = getBinding(element, type);
if (widgetInstance) {
//Destroy widget
widgetInstance.destroy();
widgetInstance.trigger("widgetdestroyed");
removeBinding(element, type);
}
}
/**
* Calls destroy on widget (or widgets) connected with given HTMLElement
* Removes child widgets as well.
* @method destroyAllWidgets
* @param {HTMLElement|string} element
* @param {boolean} [childOnly=false] destroy only widgets on children elements
* @static
* @member ns.engine
*/
function destroyAllWidgets(element, childOnly) {
var widgetName,
widgetInstance,
widgetGroup,
childWidgets,
i;
if (typeof element === TYPE_STRING) {
element = document.getElementById(element);
}
if (!childOnly) {
// If type is not defined all widgets should be removed
// this is for backward compatibility
widgetGroup = getAllBindings(element);
for (widgetName in widgetGroup) {
if (widgetGroup.hasOwnProperty(widgetName)) {
widgetInstance = widgetGroup[widgetName];
//Destroy widget
if (widgetInstance) {
widgetInstance.destroy();
widgetInstance.trigger("widgetdestroyed");
}
}
}
}
//Destroy child widgets, if something left.
childWidgets = slice.call(element.querySelectorAll("[" + DATA_BOUND + "]"));
for (i = childWidgets.length - 1; i >= 0; i -= 1) {
if (childWidgets[i]) {
destroyAllWidgets(childWidgets[i], false);
}
}
removeAllBindings(element);
}
/**
* Load widgets from data-* definition
* @method processHollowWidget
* @param {HTMLElement} element base element of widget
* @param {Object} definition widget definition
* @param {Object} [options] options for create widget
* @return {HTMLElement} base element of widget
* @private
* @static
* @member ns.engine
*/
function processHollowWidget(element, definition, options) {
var name = (element && element.getAttribute(DATA_NAME)) ||
(definition && definition.name);
definition = definition || (name && widgetDefs[name]) || {
"name": name
};
return processWidget(element, definition, options);
}
/**
* Compare function for nodes on build queue
* @param {Object} nodeA
* @param {Object} nodeB
* @return {number}
* @private
* @static
*/
function compareByDepth(nodeA, nodeB) {
var mask = Node.DOCUMENT_POSITION_CONTAINS | Node.DOCUMENT_POSITION_PRECEDING;
if (nodeA.element === nodeB.element) {
return 0;
}
if (nodeA.element.compareDocumentPosition(nodeB.element) & mask) {
return 1;
}
return -1;
}
/**
* Processes one build queue item. Runs processHollowWidget
* underneath
* @method processBuildQueueItem
* @param {Object|HTMLElement} queueItem
* @private
* @static
*/
function processBuildQueueItem(queueItem) {
// HTMLElement doesn't have .element property
// widgetDefs will return undefined when called widgetDefs[undefined]
processHollowWidget(queueItem.element || queueItem, widgetDefs[queueItem.widgetName]);
}
/**
* Build widgets on all children of context element
* @method createWidgets
* @static
* @param {HTMLElement} context base html for create children
* @member ns.engine
*/
function createWidgets(context) {
var builtWithoutTemplates = slice.call(context.querySelectorAll(querySelectorWidgets)),
normal = [],
buildQueue = [],
selectorKeys = Object.keys(widgetDefs),
excludeSelector,
i,
j,
len = selectorKeys.length,
definition,
widgetName,
definitionSelectors;
// @TODO EXPERIMENTAL WIDGETS WITHOUT TEMPLATE DEFINITION
builtWithoutTemplates.forEach(processBuildQueueItem);
/* NORMAL */
for (i = 0; i < len; ++i) {
widgetName = selectorKeys[i];
definition = widgetDefs[widgetName];
definitionSelectors = definition.selectors;
if (definitionSelectors.length) {
excludeSelector = excludeBuiltAndBound(widgetName);
normal = slice.call(context.querySelectorAll(definitionSelectors.join(excludeSelector + ",") + excludeSelector));
j = normal.length;
while (--j >= 0) {
buildQueue.push({
element: normal[j],
widgetName: widgetName
});
}
}
}
// Sort queue by depth, on every DOM branch outer most element go first
buildQueue.sort(compareByDepth);
// Build all widgets from queue
buildQueue.forEach(processBuildQueueItem);
eventUtils.trigger(document, "built");
eventUtils.trigger(document, eventType.BOUND);
}
/**
* Handler for event create
* @method createEventHandler
* @param {Event} event
* @static
* @member ns.engine
*/
function createEventHandler(event) {
createWidgets(event.target);
}
function setViewport() {
/**
* Sets viewport tag if not exists
*/
var documentHead = document.head,
metaTagListLength,
metaTagList,
metaTag,
i;
metaTagList = documentHead.querySelectorAll('[name="viewport"]');
metaTagListLength = metaTagList.length;
if (metaTagListLength > 0) {
// Leave the last viewport tag
--metaTagListLength;
// Remove duplicated tags
for (i = 0; i < metaTagListLength; ++i) {
// Remove meta tag from DOM
documentHead.removeChild(metaTagList[i]);
}
} else {
// Create new HTML Element
metaTag = document.createElement('meta');
// Set required attributes
metaTag.setAttribute('name', 'viewport');
metaTag.setAttribute('content', 'width=device-width, user-scalable=no');
// Force that viewport tag will be first child of head
if (documentHead.firstChild) {
documentHead.insertBefore(metaTag, documentHead.firstChild);
} else {
documentHead.appendChild(metaTag);
}
}
}
/**
* Build first page
* @method build
* @static
* @member ns.engine
*/
function build() {
if (router) {
// @TODO: Consider passing viewport options via script tag arguments (web-ui-fw style).
setViewport();
eventUtils.trigger(document, "beforerouterinit", router, false);
router.init(justBuild);
eventUtils.trigger(document, "routerinit", router, false);
}
}
/**
* Method to remove all listeners bound in run
* @method stop
* @static
* @member ns.engine
*/
function stop() {
if (router) {
router.destroy();
}
}
/**
* Add to object value at index equal to type of arg.
* @method getType
* @param {Object} result
* @param {*} arg
* @return {Object}
* @static
* @private
* @member ns.engine
*/
function getType(result, arg) {
var type = arg instanceof HTMLElement ? "HTMLElement" : typeof arg;
result[type] = arg;
return result;
}
/**
* Convert args array to object with keys being types and arguments mapped by values
* @method getArgumentsTypes
* @param {Arguments[]} args
* @return {Object}
* @static
* @private
* @member ns.engine
*/
function getArgumentsTypes(args) {
return tau.util.array.reduce(args, getType, {});
}
/*
document.addEventListener(eventType.BOUND, function () {
//@TODO dump it to file for faster binding by ids
nsWidgetBindingMap = widgetBindingMap;
}, false);
*/
ns.widgetDefinitions = {};
engine = {
justBuild: location.hash === "#build",
/**
* object with names of engine attributes
* @property {Object} dataTau
* @property {string} [dataTau.built="data-tau-built"] attribute inform that widget id build
* @property {string} [dataTau.name="data-tau-name"] attribute contains widget name
* @property {string} [dataTau.bound="data-tau-bound"] attribute inform that widget id bound
* @property {string} [dataTau.separator=","] separation string for widget names
* @static
* @member ns.engine
*/
dataTau: {
built: DATA_BUILT,
name: DATA_NAME,
bound: DATA_BOUND,
separator: NAMES_SEPARATOR
},
destroyWidget: destroyWidget,
destroyAllWidgets: destroyAllWidgets,
createWidgets: createWidgets,
/**
* Method to get all definitions of widgets
* @method getDefinitions
* @return {Object}
* @static
* @member ns.engine
*/
getDefinitions: function () {
return widgetDefs;
},
/**
* Returns definition of widget
* @method getWidgetDefinition
* @param {string} name
* @static
* @member ns.engine
* @returns {Object}
*/
getWidgetDefinition: function (name) {
return widgetDefs[name];
},
defineWidget: defineWidget,
getBinding: getBinding,
getAllBindings: getAllBindings,
setBinding: setBinding,
// @TODO either rename or fix functionally because
// this method does not only remove binding but
// actually destroys widget
removeBinding: removeBinding,
removeAllBindings: removeAllBindings,
/**
* Clear bindings of widgets
* @method _clearBindings
* @static
* @member ns.engine
*/
_clearBindings: function () {
//clear and set references to the same object
widgetBindingMap = {};
},
build: build,
/**
* Run engine
* @method run
* @static
* @member ns.engine
*/
run: function () {
stop();
eventUtils.fastOn(document, "create", createEventHandler);
eventUtils.trigger(document, eventType.INIT, {tau: ns});
switch (document.readyState) {
case "interactive":
case "complete":
build();
break;
default:
eventUtils.fastOn(document, "DOMContentLoaded", build.bind(engine));
break;
}
},
/**
* Return router
* @method getRouter
* @return {Object}
* @static
* @member ns.engine
*/
getRouter: function () {
return router;
},
/**
* Initialize router. This method should be call in file with router class definition.
* @method initRouter
* @param {Function} RouterClass Router class
* @static
* @member ns.engine
*/
initRouter: function (RouterClass) {
router = new RouterClass();
},
/**
* Build instance of widget and binding events
* Returns error when empty element is passed
* @method instanceWidget
* @param {HTMLElement} [element]
* @param {string} name
* @param {Object} [options]
* @return {?Object}
* @static
* @member ns.engine
*/
instanceWidget: function (element, name, options) {
var binding,
definition,
argumentsTypes = getArgumentsTypes(arguments);
// Map arguments with specific types to correct variables
// Only name is required argument
element = argumentsTypes.HTMLElement;
name = argumentsTypes.string;
options = argumentsTypes.object;
// If element exists try to find existing binding
if (element) {
binding = getBinding(element, name);
}
// If didn't found binding build new widget
if (!binding && widgetDefs[name]) {
definition = widgetDefs[name];
element = processHollowWidget(element, definition, options);
binding = getBinding(element, name);
}
return binding;
},
stop: stop,
/**
* Method to change build mode
* @method setJustBuild
* @param {boolean} newJustBuild
* @static
* @member ns.engine
*/
setJustBuild: function (newJustBuild) {
// Set location hash to have a consistent behavior
if(newJustBuild){
location.hash = "build";
} else {
location.hash = "";
}
justBuild = newJustBuild;
},
/**
* Method to get build mode
* @method getJustBuild
* @return {boolean}
* @static
* @member ns.engine
*/
getJustBuild: function () {
return justBuild;
},
_createEventHandler : createEventHandler
};
engine.eventType = eventType;
ns.engine = engine;
}(window, window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Anchor Highlight Utility
* Utility enables highlight links.
* @class ns.util.anchorHighlight
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Damian Osipiuk <d.osipiuk@samsung.com>
* @author Konrad Lipner <k.lipner@samsung.com>
*/
(function (document, window, ns) {
"use strict";
/* anchorHighlightController.js
To prevent perfomance regression when scrolling,
do not apply hover class in anchor.
Instead, this code checks scrolling for time threshold and
decide how to handle the color.
When scrolling with anchor, it checks flag and decide to highlight anchor.
While it helps to improve scroll performance,
it lowers responsiveness of the element for 50msec.
*/
/**
* Touch start x
* @property {number} startX
* @member ns.util.anchorHighlight
* @private
* @static
*/
var startX = 0,
/**
* Touch start y
* @property {number} startY
* @member ns.util.anchorHighlight
* @private
* @static
*/
startY = 0,
/**
* Did page scrolled
* @property {boolean} didScroll
* @member ns.util.anchorHighlight
* @private
* @static
*/
didScroll = false,
/**
* Touch target element
* @property {HTMLElement} target
* @member ns.util.anchorHighlight
* @private
* @static
*/
target = null,
/**
* Object with default options
* @property {Object} options
* Treshold after which didScroll will be set
* @property {number} [options.scrollThreshold=30]
* Time to wait before adding activeClass
* @property {number} [options.addActiveClassDelay=10]
* Time to stay activeClass after touch end
* @property {number} [options.keepActiveClassDelay=100]
* @member ns.util.anchorHighlight
* @private
* @static
*/
options = {
scrollThreshold: 30,
addActiveClassDelay: 10,
keepActiveClassDelay: 100
},
/**
* Class used to mark element as active
* @property {string} [activeClassLI="ui-li-active"] activeClassLI
* @member ns.util.anchorHighlight
* @private
* @static
*/
activeClassLI = "ui-li-active",
/**
* Alias for class {@link ns.util.selectors}
* @property {Object} selectors
* @member ns.util.anchorHighlight
* @private
* @static
*/
selectors = ns.util.selectors,
// cache function
requestAnimationFrame = ns.util.requestAnimationFrame,
// cache function
abs = Math.abs,
startTime = 0,
startRemoveTime = 0,
// cache function
slice = Array.prototype.slice;
/**
* Get closest highlightable element
* @method detectHighlightTarget
* @param {HTMLElement} target
* @return {HTMLElement}
* @member ns.util.anchorHighlight
* @private
* @static
*/
function detectHighlightTarget(target) {
return selectors.getClosestBySelector(target, "a, label");
}
/**
* Get closest li element
* @method detectLiElement
* @param {HTMLElement} target
* @return {HTMLElement}
* @member ns.util.anchorHighlight
* @private
* @static
*/
function detectLiElement(target) {
return selectors.getClosestByTag(target, "li");
}
/**
* Add active class to touched element
* @method addActiveClass
* @member ns.util.anchorHighlight
* @private
* @static
*/
function addActiveClass() {
var liTarget = null,
dTime = 0;
if (startTime) {
dTime = Date.now() - startTime;
if (dTime > options.addActiveClassDelay) {
startTime = 0;
target = detectHighlightTarget(target);
if (!didScroll) {
liTarget = detectLiElement(target);
if( liTarget ) {
liTarget.classList.add(activeClassLI);
}
liTarget = null;
}
} else {
requestAnimationFrame(addActiveClass);
}
}
}
/**
* Get all active elements
* @method getActiveElements
* @return {Array}
* @member ns.util.anchorHighlight
* @private
* @static
*/
function getActiveElements() {
return slice.call(document.getElementsByClassName(activeClassLI));
}
/**
* Remove active class from current active objects
*/
function clearActiveClass () {
var activeA = getActiveElements(),
activeALength = activeA.length,
i = 0;
for (; i < activeALength; i++) {
activeA[i].classList.remove(activeClassLI);
}
activeA = null;
}
/**
* Remove active class from active elements
* @method removeActiveClass
* @member ns.util.anchorHighlight
* @private
* @static
*/
function removeActiveClass() {
var dTime = Date.now() - startRemoveTime;;
if (dTime > options.keepActiveClassDelay) {
// after touchend
clearActiveClass();
} else {
requestAnimationFrame(removeActiveClass);
}
}
/**
* Function invoked during touch move
* @method touchmoveHandler
* @param {Event} event
* @member ns.util.anchorHighlight
* @private
* @static
*/
function touchmoveHandler(event) {
var touch = event.touches[0],
scrollThreshold = options.scrollThreshold;
// if move looks like scroll
if (!didScroll &&
// if move is bigger then threshold
(abs(touch.clientX - startX) > scrollThreshold || abs(touch.clientY - startY) > scrollThreshold)) {
startTime = 0;
// we clear active classes
requestAnimationFrame(clearActiveClass);
didScroll = true;
}
touch = null;
}
/**
* Function invoked after touch start
* @method touchstartHandler
* @param {Event} event
* @member ns.util.anchorHighlight
* @private
* @static
*/
function touchstartHandler(event) {
var touches = event.touches,
touch = null;
if (touches.length === 1) {
touch = touches[0];
didScroll = false;
startX = touch.clientX;
startY = touch.clientY;
target = event.target;
startTime = event.timeStamp;
startRemoveTime = 0;
requestAnimationFrame(addActiveClass);
touch = null;
}
touches = null;
}
/**
* Function invoked after touch
* @method touchendHandler
* @param {Event} event
* @member ns.util.anchorHighlight
* @private
* @static
*/
function touchendHandler(event) {
startRemoveTime = event.timeStamp;
if (event.touches.length === 0) {
if (!didScroll) {
startTime = 0;
requestAnimationFrame(removeActiveClass);
}
didScroll = false;
}
}
/**
* Function invoked after visibilitychange event
* @method checkPageVisibility
* @member ns.util.anchorHighlight
* @private
* @static
*/
function checkPageVisibility() {
if (document.visibilityState === "hidden") {
removeActiveClass();
}
}
/**
* Bind events to document
* @method enable
* @member ns.util.anchorHighlight
* @static
*/
function enable() {
document.addEventListener("touchstart", touchstartHandler, false);
document.addEventListener("touchend", touchendHandler, false);
document.addEventListener("touchmove", touchmoveHandler, false);
document.addEventListener("visibilitychange", checkPageVisibility, false);
window.addEventListener("pagehide", removeActiveClass, false);
}
/**
* Unbinds events from document.
* @method disable
* @member ns.util.anchorHighlight
* @static
*/
function disable() {
document.removeEventListener("touchstart", touchstartHandler, false);
document.removeEventListener("touchend", touchendHandler, false);
document.removeEventListener("touchmove", touchmoveHandler, false);
document.removeEventListener("visibilitychange", checkPageVisibility, false);
window.removeEventListener("pagehide", removeActiveClass, false);
}
enable();
ns.util.anchorHighlight = {
enable: enable,
disable: disable
};
}(document, window, ns));
/* global requestAnimationFrame, define, ns */
/**
* # JS base scrolling tool
*
* This enable fast scrolling on element
*
* @class ns.util.scrolling
*/
(function (document, window, ns) {
"use strict";
var eventUtil = ns.event,
// position when was last touch start
startPosition = 0,
// current state of scroll position
scrollPosition = 0,
lastScrollPosition = 0,
moveToPosition = 0,
lastRenderedPosition = 0,
lastTime = Date.now(),
elementStyle = null,
maxScrollPosition = 0,
// scrolling element
scollingElement = null,
// cache of previous overflow style to revert after disable
previousOverflow = "",
// cache abs function
abs = Math.abs,
// inform that is touched
isTouch = false,
isScoll = false,
// direction of scrolling, 0 - mean Y, 1 - mean X
direction = 0,
// cache of round function
round = Math.round;
/**
* Chack that current target is inside scrolling element
* @param {HTMLElement} target
* @return boolean
*/
function detectTarget(target) {
while (target && target !== document) {
if (target === scollingElement) {
return true;
}
target = target.parentElement;
}
return false;
}
/**
* Handler for touchstart event
* @param {Event} event
*/
function touchStart(event) {
var touches = event.touches,
touch = touches[0];
isScoll = detectTarget(event.target);
// is is only one touch
if (isScoll && touches.length === 1) {
// save current touch point
startPosition = direction ? touch.clientX : touch.clientY;
// save current time for calculate acceleration on touchend
lastTime = Date.now();
// reset acceleration state
moveToPosition = scrollPosition;
}
// clean
touches = null;
touch = null;
}
/**
* Handler for touchmove event
* @param event
*/
function touchMove(event) {
var touches = event.touches,
touch = touches[0],
// get current position in correct direction
clientPosition = direction ? touch.clientX : touch.clientY;
// if touch start was on scrolled element
if (isScoll) {
// if is only one touch
if (touches.length === 1) {
// calculate difference between touch start and current position
lastScrollPosition = clientPosition - startPosition;
// normalize value to be in bound [0, maxScroll]
if (scrollPosition + lastScrollPosition > 0) {
lastScrollPosition = -scrollPosition;
}
if (scrollPosition + lastScrollPosition < -maxScrollPosition) {
lastScrollPosition = -maxScrollPosition - scrollPosition;
}
// trigger event scroll
eventUtil.trigger(scollingElement, "scroll", {scrollTop: -(scrollPosition + lastScrollPosition)});
}
// if this is first touch move
if (!isTouch) {
// we need start request loop
isTouch = true;
requestAnimationFrame(render);
}
}
// clean
touches = null;
touch = null;
}
/**
* Handler for touchend event
*/
function touchEnd() {
var diffTime = Date.now() - lastTime;
// calculate speed of touch move
if (abs(lastScrollPosition / diffTime) > 1) {
// if it was fast move, we start animation of scrolling after touch end
moveToPosition = round(scrollPosition + 1000 * lastScrollPosition / diffTime);
requestAnimationFrame(moveTo);
} else {
// touch move was slow, just finish render loop
isTouch = false;
}
// update state of scrolling
scrollPosition += lastScrollPosition;
// normalize value to be in bound [0, maxScroll]
if (scrollPosition < -maxScrollPosition) {
scrollPosition = -maxScrollPosition;
}
if (scrollPosition > 0) {
scrollPosition = 0;
}
lastScrollPosition = 0;
// trigger event scroll
eventUtil.trigger(scollingElement, "scroll", {scrollTop: -(scrollPosition)});
// we stop scrolling
isScoll = false;
}
/**
* Loop function to calculate state in animation after touchend
*/
function moveTo() {
// calculate difference between current position and expected scroll end
var diffPosition = moveToPosition - scrollPosition,
// get absolute value
absDiffPosition = abs(diffPosition);
// if difference is big
if (absDiffPosition > 10) {
// we move 10% of difference
scrollPosition = round(scrollPosition + diffPosition / 10);
requestAnimationFrame(moveTo);
} else if (absDiffPosition > 2) {
// else if is difference < 10 then we move 50%
scrollPosition = round(scrollPosition + diffPosition / 2);
requestAnimationFrame(moveTo);
} else {
// if difference is <=2 then we move to end value and finish loop
scrollPosition = moveToPosition;
isTouch = false;
}
// normalize scroll value
if (scrollPosition < -maxScrollPosition) {
scrollPosition = -maxScrollPosition;
}
if (scrollPosition > 0) {
scrollPosition = 0;
}
// trigger event scroll
eventUtil.trigger(scollingElement, "scroll", {scrollTop: -(scrollPosition)});
}
/**
* Render loop on request animation frame
*/
function render() {
// calculate ne position of scrolling as sum of last scrolling state + move
var newRenderedPosition = scrollPosition + lastScrollPosition;
// is position was changed
if (newRenderedPosition !== lastRenderedPosition) {
// we update styles
lastRenderedPosition = newRenderedPosition;
elementStyle.transform = direction ?
"translate3D(" + lastRenderedPosition + "px,0,0)" :
"translate3D(0, " + lastRenderedPosition + "px,0)";
}
// if is still touched then we continue loop
if (elementStyle && isTouch) {
requestAnimationFrame(render);
}
}
/**
* Enable JS scrolling on element
* @method enable
* @param {HTMLElement} element element for scrolling
* @param {"x"|"y"} [setDirection="y"] direction of scrolling
* @member ns.util.scrolling
*/
function enable(element, setDirection) {
var parentRectangle = null,
rectangle = null,
childElement = element.firstElementChild;
if (scollingElement) {
console.warn("Scrolling exist on another element, first call disable method");
} else {
// detect direction
if (setDirection === "x") {
direction = 1;
} else {
direction = 0;
}
// setting scrolling element
scollingElement = element;
// calculate maxScroll
parentRectangle = element.getBoundingClientRect();
rectangle = childElement.getBoundingClientRect();
if (direction) {
maxScrollPosition = round(rectangle.width - parentRectangle.width);
} else {
maxScrollPosition = round(rectangle.height - parentRectangle.height);
}
// cache style element
elementStyle = childElement.style;
// init internal variables
startPosition = 0;
scrollPosition = 0;
lastScrollPosition = 0;
moveToPosition = 0;
lastRenderedPosition = 0;
lastTime = Date.now();
// cache current overflow value to restore in disable
previousOverflow = window.getComputedStyle(element).getPropertyValue("overflow");
// set overflow hidden
element.style.overflow = "hidden";
// add event listeners
document.addEventListener("touchstart", touchStart, false);
document.addEventListener("touchmove", touchMove, false);
document.addEventListener("touchend", touchEnd, false);
}
}
/**
* @method disable
* @member ns.util.scrolling
*/
function disable() {
// clear event listeners
document.removeEventListener("touchstart", touchStart, false);
document.removeEventListener("touchmove", touchMove, false);
document.removeEventListener("touchend", touchEnd, false);
scollingElement.style.overflow = previousOverflow;
elementStyle = null;
scollingElement = null;
}
/**
* Scroll to give position
* @method scrollTo
* @param {number} value
* @member ns.util.scrolling
*/
function scrollTo(value) {
scrollPosition = value;
lastScrollPosition = 0;
render();
}
ns.util.scrolling = {
enable: enable,
disable: disable,
scrollTo: scrollTo,
/**
* Return true is given element is current scrolling element
* @method isElement
* @param {HTMLElement} element element to check
* @returns {boolean}
* @member ns.util.scrolling
*/
isElement: function(element) {
return scollingElement === element;
},
/**
* Update max scrolling position
* @method setMaxScroll
* @param maxValue
* @member ns.util.scrolling
*/
setMaxScroll: function(maxValue) {
maxScrollPosition = maxValue;
}
};
}(document, window, ns));
/*global window, define */
/*jslint plusplus: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Utility DOM
* Utility object with function to DOM manipulation, CSS properties support
* and DOM attributes support.
*
* # How to replace jQuery methods by ns methods
* ## append vs appendNodes
*
* #### HTML code before manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <div id="second">And</div>
* <div id="third">Goodbye</div>
* </div>
*
* #### jQuery manipulation
*
* @example
* $( "#second" ).append( "<span>Test</span>" );
* #### ns manipulation
*
* @example
* var context = document.getElementById("second"),
* element = document.createElement("span");
* element.innerHTML = "Test";
* ns.util.DOM.appendNodes(context, element);
*
* #### HTML code after manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <div id="second">And
* <span>Test</span>
* </div>
* <div id="third">Goodbye</div>
* </div>
*
* ## replaceWith vs replaceWithNodes
*
* #### HTML code before manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <div id="second">And</div>
* <div id="third">Goodbye</div>
* </div>
*
* #### jQuery manipulation
*
* @example
* $('#second').replaceWith("<span>Test</span>");
*
* #### ns manipulation
*
* @example
* var context = document.getElementById("second"),
* element = document.createElement("span");
* element.innerHTML = "Test";
* ns.util.DOM.replaceWithNodes(context, element);
*
* #### HTML code after manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <span>Test</span>
* <div id="third">Goodbye</div>
* </div>
*
* ## before vs insertNodesBefore
*
* #### HTML code before manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <div id="second">And</div>
* <div id="third">Goodbye</div>
* </div>
*
* #### jQuery manipulation
*
* @example
* $( "#second" ).before( "<span>Test</span>" );
*
* #### ns manipulation
*
* @example
* var context = document.getElementById("second"),
* element = document.createElement("span");
* element.innerHTML = "Test";
* ns.util.DOM.insertNodesBefore(context, element);
*
* #### HTML code after manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <span>Test</span>
* <div id="second">And</div>
* <div id="third">Goodbye</div>
* </div>
*
* ## wrapInner vs wrapInHTML
*
* #### HTML code before manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <div id="second">And</div>
* <div id="third">Goodbye</div>
* </div>
*
* #### jQuery manipulation
*
* @example
* $( "#second" ).wrapInner( "<span class="new"></span>" );
*
* #### ns manipulation
*
* @example
* var element = document.getElementById("second");
* ns.util.DOM.wrapInHTML(element, "<span class="new"></span>");
*
* #### HTML code after manipulation
*
* @example
* <div>
* <div id="first">Hello</div>
* <div id="second">
* <span class="new">And</span>
* </div>
* <div id="third">Goodbye</div>
* </div>
*
* @class ns.util.DOM
* @author Jadwiga Sosnowska <j.sosnowska@partner.samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (ns) {
"use strict";
ns.util.DOM = ns.util.DOM || {};
}(ns));
/*global window, define */
/*jslint plusplus: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @author Jadwiga Sosnowska <j.sosnowska@partner.samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (window, document, ns) {
"use strict";
var selectors = ns.util.selectors,
DOM = ns.util.DOM,
namespace = "namespace";
/**
* Returns given attribute from element or the closest parent,
* which matches the selector.
* @method inheritAttr
* @member ns.util.DOM
* @param {HTMLElement} element
* @param {string} attr
* @param {string} selector
* @return {?string}
* @static
*/
DOM.inheritAttr = function (element, attr, selector) {
var value = element.getAttribute(attr),
parent;
if (!value) {
parent = selectors.getClosestBySelector(element, selector);
if (parent) {
return parent.getAttribute(attr);
}
}
return value;
};
/**
* Returns Number from properties described in html tag
* @method getNumberFromAttribute
* @member ns.util.DOM
* @param {HTMLElement} element
* @param {string} attribute
* @param {string=} [type] auto type casting
* @param {number} [defaultValue] default returned value
* @static
* @return {number}
*/
DOM.getNumberFromAttribute = function (element, attribute, type, defaultValue) {
var value = element.getAttribute(attribute),
result = defaultValue;
if (!isNaN(value)) {
if (type === "float") {
value = parseFloat(value);
if (!isNaN(value)) {
result = value;
}
} else {
value = parseInt(value, 10);
if (!isNaN(value)) {
result = value;
}
}
}
return result;
};
function getDataName(name) {
var namespace = ns.getConfig(namespace);
return "data-" + (namespace ? namespace + "-" : "") + name;
}
/**
* Special function to set attribute and property in the same time
* @method setAttribute
* @param {HTMLElement} element
* @param {string} name
* @param {Mixed} value
* @member ns.util.DOM
* @static
*/
function setAttribute(element, name, value) {
element[name] = value;
element.setAttribute(name, value);
}
/**
* This function sets value of attribute data-{namespace}-{name} for element.
* If the namespace is empty, the attribute data-{name} is used.
* @method setNSData
* @param {HTMLElement} element Base element
* @param {string} name Name of attribute
* @param {string|number|boolean} value New value
* @member ns.util.DOM
* @static
*/
DOM.setNSData = function (element, name, value) {
element.setAttribute(getDataName(name), value);
};
/**
* This function returns value of attribute data-{namespace}-{name} for element.
* If the namespace is empty, the attribute data-{name} is used.
* Method may return boolean in case of 'true' or 'false' strings as attribute value.
* @method getNSData
* @param {HTMLElement} element Base element
* @param {string} name Name of attribute
* @member ns.util.DOM
* @return {?string|boolean}
* @static
*/
DOM.getNSData = function (element, name) {
var value = element.getAttribute(getDataName(name));
if (value === "true") {
return true;
}
if (value === "false") {
return false;
}
return value;
};
/**
* This function returns true if attribute data-{namespace}-{name} for element is set
* or false in another case. If the namespace is empty, attribute data-{name} is used.
* @method hasNSData
* @param {HTMLElement} element Base element
* @param {string} name Name of attribute
* @member ns.util.DOM
* @return {boolean}
* @static
*/
DOM.hasNSData = function (element, name) {
return element.hasAttribute(getDataName(name));
};
/**
* Get or set value on data attribute.
* @method nsData
* @param {HTMLElement} element
* @param {string} name
* @param {?Mixed} [value]
* @static
* @member ns.util.DOM
*/
DOM.nsData = function (element, name, value) {
// @TODO add support for object in value
if (value === undefined) {
return DOM.getNSData(element, name);
} else {
return DOM.setNSData(element, name, value);
}
};
/**
* This function removes attribute data-{namespace}-{name} from element.
* If the namespace is empty, attribute data-{name} is used.
* @method removeNSData
* @param {HTMLElement} element Base element
* @param {string} name Name of attribute
* @member ns.util.DOM
* @static
*/
DOM.removeNSData = function (element, name) {
element.removeAttribute(getDataName(name));
};
/**
* Returns object with all data-* attributes of element
* @method getData
* @param {HTMLElement} element Base element
* @member ns.util.DOM
* @return {Object}
* @static
*/
DOM.getData = function (element) {
var dataPrefix = "data-",
data = {},
attrs = element.attributes,
attr,
nodeName,
value,
i,
length = attrs.length;
for (i = 0; i < length; i++) {
attr = attrs.item(i);
nodeName = attr.nodeName;
if (nodeName.indexOf(dataPrefix) > -1) {
value = attr.value;
data[nodeName.replace(dataPrefix, "")] = value.toLowerCase() === "true" ? true : value.toLowerCase() === "false" ? false : value;
}
}
return data;
};
/**
* Special function to remove attribute and property in the same time
* @method removeAttribute
* @param {HTMLElement} element
* @param {string} name
* @member ns.util.DOM
* @static
*/
DOM.removeAttribute = function (element, name) {
element.removeAttribute(name);
element[name] = false;
};
DOM.setAttribute = setAttribute;
/**
* Special function to set attributes and propertie in the same time
* @method setAttribute
* @param {HTMLElement} element
* @param {Object} name
* @param {Mixed} value
* @member ns.util.DOM
* @static
*/
DOM.setAttributes = function (element, values) {
var i,
names = Object.keys(values),
name,
len;
for (i = 0, len = names.length; i < len; i++) {
name = names[i];
setAttribute(element, name, values[name]);
}
};
}(window, window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Namespace For Widgets
* Namespace For Widgets
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @class ns.widget
*/
(function (document, ns) {
"use strict";
var engine = ns.engine,
widget = {
/**
* Get bound widget for element
* @method getInstance
* @static
* @param {HTMLElement|string} element
* @param {string} type widget name
* @return {?Object}
* @member ns.widget
*/
getInstance: engine.getBinding,
/**
* Returns Get all bound widget for element or id gives as parameter
* @method getAllInstances
* @param {HTMLElement|string} element
* @return {?Object}
* @static
* @member ns.widget
*/
getAllInstances: engine.getAllBindings
};
function widgetConstructor(name, element, options) {
return engine.instanceWidget(element, name, options);
}
document.addEventListener(engine.eventType.WIDGET_DEFINED, function (evt) {
var definition = evt.detail,
name = definition.name;
ns.widget[name] = widgetConstructor.bind(null, name);
}, true);
/** @namespace ns.widget */
ns.widget = widget;
}(window.document, ns));
/*global window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/*
*/
/**
* #BaseWidget
* Prototype class of widget
*
* ## How to invoke creation of widget from JavaScript
*
* To build and initialize widget in JavaScript you have to use method {@link ns.engine#instanceWidget} . First argument for method
* is HTMLElement, which specifies the element of widget. Second parameter is name of widget to create.
*
* If you load jQuery before initializing tau library, you can use standard jQuery UI Widget notation.
*
* ### Examples
* #### Build widget from JavaScript
*
* @example
* var element = document.getElementById("id"),
* ns.engine.instanceWidget(element, "Button");
*
* #### Build widget from jQuery
*
* @example
* var element = $("#id").button();
*
* ## How to create new widget
*
* @example
* (function (ns) {
* "use strict";
* * var BaseWidget = ns.widget.BaseWidget, // create alias to main objects
* ...
* arrayOfElements, // example of private property, common for all instances of widget
* Button = function () { // create local object with widget
* ...
* },
* prototype = new BaseWidget(); // add ns.widget.BaseWidget as prototype to widget's object, for better minification this should be assign to local variable and next variable should be assign to prototype of object
*
* function closestEnabledButton(element) { // example of private method
* ...
* }
* ...
*
* prototype.options = { //add default options to be read from data- attributes
* theme: "s",
* ...
* };
*
* prototype._build = function (template, element) { // method called when the widget is being built, should contain all HTML manipulation actions
* ...
* return element;
* };
*
* prototype._init = function (element) { // method called during initialization of widget, should contain all actions necessary fastOn application start
* ...
* return element;
* };
*
* prototype._bindEvents = function (element) { // method to bind all events, should contain all event bindings
* ...
* };
*
* prototype._enable = function (element) { // method called during invocation of enable() method
* ...
* };
*
* prototype._disable = function (element) { // method called during invocation of disable() method
* ...
* };
*
* prototype.refresh = function (element) { // example of public method
* ...
* };
*
* prototype._refresh = function () { // example of protected method
* ...
* };
*
* Button.prototype = prototype;
*
* engine.defineWidget( // define widget
* "Button", //name of widget
* "[data-role='button'],button,[type='button'],[type='submit'],[type='reset']", //widget's selector
* [ // public methods, here should be list all public method, without that method will not be available
* "enable",
* "disable",
* "refresh"
* ],
* Button, // widget's object
* "mobile" // widget's namespace
* );
* ns.widget.Button = Button;
* * }(ns));
* @author Jadwiga Sosnowska <j.sosnowska@samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Tomasz Lukawski <t.lukawski@samsung.com>
* @author Przemyslaw Ciezkowski <p.ciezkowski@samsung.com>
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
* @author Michał Szepielak <m.szepielak@samsung.com>
* @class ns.widget.BaseWidget
*/
(function (document, ns, undefined) {
"use strict";
/**
* Alias to Array.slice function
* @method slice
* @member ns.widget.BaseWidget
* @private
* @static
*/
var slice = [].slice,
/**
* Alias to ns.engine
* @property {ns.engine} engine
* @member ns.widget.BaseWidget
* @private
* @static
*/
engine = ns.engine,
engineDataTau = engine.dataTau,
util = ns.util,
/**
* Alias to {@link ns.event}
* @property {Object} eventUtils
* @member ns.widget.BaseWidget
* @private
* @static
*/
eventUtils = ns.event,
/**
* Alias to {@link ns.util.DOM}
* @property {Object} domUtils
* @private
* @static
*/
domUtils = util.DOM,
/**
* Alias to {@link ns.util.object}
* @property {Object} objectUtils
* @private
* @static
*/
objectUtils = util.object,
BaseWidget = function () {
return this;
},
prototype = {},
/**
* Property with string represent function type
* (for better minification)
* @property {string} [TYPE_FUNCTION="function"]
* @private
* @static
* @readonly
*/
TYPE_FUNCTION = "function",
disableClass = "ui-state-disabled",
ariaDisabled = "aria-disabled";
BaseWidget.classes = {
disable: disableClass
};
/**
* Protected method configuring the widget
* @method _configure
* @member ns.widget.BaseWidget
* @protected
* @template
* @internal
*/
/**
* Configures widget object from definition.
*
* It calls such methods as #\_getCreateOptions and #\_configure.
* @method configure
* @param {Object} definition
* @param {string} definition.name Name of the widget
* @param {string} definition.selector Selector of the widget
* @param {HTMLElement} element Element of widget
* @param {Object} options Configure options
* @member ns.widget.BaseWidget
* @chainable
* @internal
*/
prototype.configure = function (definition, element, options) {
var self = this,
definitionName,
definitionNamespace;
/**
* Object with options for widget
* @property {Object} [options={}]
* @member ns.widget.BaseWidget
*/
self.options = self.options || {};
/**
* Base element of widget
* @property {?HTMLElement} [element=null]
* @member ns.widget.BaseWidget
*/
self.element = self.element || null;
if (definition) {
definitionName = definition.name;
definitionNamespace = definition.namespace;
/**
* Name of the widget
* @property {string} name
* @member ns.widget.BaseWidget
*/
self.name = definitionName;
/**
* Name of the widget (in lower case)
* @property {string} widgetName
* @member ns.widget.BaseWidget
*/
self.widgetName = definitionName;
/**
* Namespace of widget events
* @property {string} widgetEventPrefix
* @member ns.widget.BaseWidget
*/
self.widgetEventPrefix = definitionName.toLowerCase();
/**
* Namespace of the widget
* @property {string} namespace
* @member ns.widget.BaseWidget
*/
self.namespace = definitionNamespace;
/**
* Full name of the widget
* @property {string} widgetFullName
* @member ns.widget.BaseWidget
*/
self.widgetFullName = ((definitionNamespace ? definitionNamespace + "-" : "") + definitionName).toLowerCase();
/**
* Id of widget instance
* @property {string} id
* @member ns.widget.BaseWidget
*/
self.id = ns.getUniqueId();
/**
* Widget's selector
* @property {string} selector
* @member ns.widget.BaseWidget
*/
self.selector = definition.selector;
}
if (typeof self._configure === TYPE_FUNCTION) {
self._configure(element);
}
self._getCreateOptions(element);
objectUtils.fastMerge(self.options, options);
};
/**
* Reads data-* attributes and save to options object.
* @method _getCreateOptions
* @param {HTMLElement} element Base element of the widget
* @return {Object}
* @member ns.widget.BaseWidget
* @protected
*/
prototype._getCreateOptions = function (element) {
var options = this.options,
bigRegexp = /[A-Z]/g;
if (options !== undefined) {
Object.keys(options).forEach(function (option) {
// Get value from data-{namespace}-{name} element's attribute
// based on widget.options property keys
var value = domUtils.getNSData(element, (option.replace(bigRegexp, function (c) {
return "-" + c.toLowerCase();
})));
if (value !== null) {
options[option] = value;
}
});
}
return options;
};
/**
* Protected method building the widget
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement} widget's element
* @member ns.widget.BaseWidget
* @protected
* @template
*/
/**
* Builds widget.
*
* It calls method #\_build.
*
* Before starting building process, the event beforecreate with
* proper prefix defined in variable widgetEventPrefix is triggered.
* @method build
* @param {HTMLElement} element Element of widget before building process
* @return {HTMLElement} Element of widget after building process
* @member ns.widget.BaseWidget
* @internal
*/
prototype.build = function (element) {
var self = this,
id,
node,
dataBuilt = element.getAttribute(engineDataTau.built),
dataName = element.getAttribute(engineDataTau.name);
eventUtils.trigger(element, self.widgetEventPrefix + "beforecreate");
id = element.id;
if (id) {
self.id = id;
} else {
element.id = self.id;
}
if (typeof self._build === TYPE_FUNCTION) {
node = self._build(element);
} else {
node = element;
}
// Append current widget name to data-tau-built and data-tau-name attributes
dataBuilt = !dataBuilt ? self.name : dataBuilt + engineDataTau.separator + self.name;
dataName = !dataName ? self.name : dataName + engineDataTau.separator + self.name;
element.setAttribute(engineDataTau.built, dataBuilt);
element.setAttribute(engineDataTau.name, dataName);
return node;
};
/**
* Protected method initializing the widget
* @method _init
* @param {HTMLElement} element
* @member ns.widget.BaseWidget
* @template
* @protected
*/
/**
* Initializes widget.
*
* It calls method #\_init.
* @method init
* @param {HTMLElement} element Element of widget before initialization
* @member ns.widget.BaseWidget
* @chainable
* @internal
*/
prototype.init = function (element) {
var self = this;
self.id = element.id;
if (typeof self._init === TYPE_FUNCTION) {
self._init(element);
}
if (element.getAttribute("disabled") || self.options.disabled === true) {
self.disable();
} else {
self.enable();
}
return self;
};
/**
* Returns base element widget
* @member ns.widget.BaseWidget
* @return {HTMLElement|null}
* @instance
*/
prototype.getContainer = function () {
var self = this;
if (typeof self._getContainer === TYPE_FUNCTION) {
return self._getContainer();
}
return self.element;
};
/**
* Bind widget events attached in init mode
* @method _bindEvents
* @param {HTMLElement} element Base element of widget
* @member ns.widget.BaseWidget
* @template
* @protected
*/
/**
* Binds widget events.
*
* It calls such methods as #\_buildBindEvents and #\_bindEvents.
* At the end of binding process, the event "create" with proper
* prefix defined in variable widgetEventPrefix is triggered.
* @method bindEvents
* @param {HTMLElement} element Base element of the widget
* @param {boolean} onlyBuild Inform about the type of bindings: build/init
* @member ns.widget.BaseWidget
* @chainable
* @internal
*/
prototype.bindEvents = function (element, onlyBuild) {
var self = this,
dataBound = element.getAttribute(engineDataTau.bound);
if (!onlyBuild) {
dataBound = !dataBound ? self.name : dataBound + engineDataTau.separator + self.name;
element.setAttribute(engineDataTau.bound, dataBound);
}
if (typeof self._buildBindEvents === TYPE_FUNCTION) {
self._buildBindEvents(element);
}
if (!onlyBuild && typeof self._bindEvents === TYPE_FUNCTION) {
self._bindEvents(element);
}
self.trigger(self.widgetEventPrefix + "create", self);
return self;
};
/**
* Focus widget's element.
*
* This function calls function focus on element and if it is known
* the direction of event, the proper css classes are added/removed.
* @method focus
* @param {object} options The options of event.
* @param {"up"|"down"|"left"|"right"} direction
* For example, if this parameter has value "down", it means that the movement
* comes from the top (eg. down arrow was pressed on keyboard).
* @param {HTMLElement} previousElement Element to blur
* @member ns.widget.BaseWidget
*/
prototype.focus = function (options) {
var self = this,
element = self.element,
blurElement,
blurWidget;
options = options || {};
blurElement = options.previousElement;
// we try to blur element, which has focus previously
if (blurElement) {
blurWidget = engine.getBinding(blurElement);
// call blur function on widget
if (blurWidget) {
options = objectUtils.merge({}, options, {element: blurElement});
blurWidget.blur(options);
} else {
// or on element, if widget does not exist
blurElement.blur();
}
}
options = objectUtils.merge({}, options, {element: element});
// set focus on element
eventUtils.trigger(document, "taufocus", options);
element.focus();
return true;
};
/**
* Blur widget's element.
*
* This function calls function blur on element and if it is known
* the direction of event, the proper css classes are added/removed.
* @method blur
* @param {object} options The options of event.
* @param {"up"|"down"|"left"|"right"} direction
* @member ns.widget.BaseWidget
*/
prototype.blur = function (options) {
var self = this,
element = self.element;
options = objectUtils.merge({}, options, {element: element});
// blur element
eventUtils.trigger(document, "taublur", options);
element.blur();
return true;
};
/**
* Protected method destroying the widget
* @method _destroy
* @template
* @protected
* @member ns.widget.BaseWidget
*/
/**
* Destroys widget.
*
* It calls method #\_destroy.
*
* At the end of destroying process, the event "destroy" with proper
* prefix defined in variable widgetEventPrefix is triggered and
* the binding set in engine is removed.
* @method destroy
* @param {HTMLElement} element Base element of the widget
* @member ns.widget.BaseWidget
*/
prototype.destroy = function (element) {
var self = this;
element = element || self.element;
if (typeof self._destroy === TYPE_FUNCTION) {
self._destroy(element);
}
if (self.element) {
self.trigger(self.widgetEventPrefix + "destroy");
}
if (element) {
engine.removeBinding(element, self.name);
}
};
/**
* Protected method disabling the widget
* @method _disable
* @protected
* @member ns.widget.BaseWidget
* @template
*/
/**
* Disables widget.
*
* It calls method #\_disable.
* @method disable
* @member ns.widget.BaseWidget
* @chainable
*/
prototype.disable = function () {
var self = this,
args = slice.call(arguments),
element = self.element;
element.classList.add(disableClass);
element.setAttribute(ariaDisabled, true);
if (typeof self._disable === TYPE_FUNCTION) {
args.unshift(element);
self._disable.apply(self, args);
}
return this;
};
/**
* Check if widget is disabled.
* @method isDisabled
* @member ns.widget.BaseWidget
* @return {boolean} Returns true if widget is disabled
*/
prototype.isDisabled = function () {
var self = this;
return self.element.getAttribute("disabled") || self.options.disabled === true;
};
/**
* Protected method enabling the widget
* @method _enable
* @protected
* @member ns.widget.BaseWidget
* @template
*/
/**
* Enables widget.
*
* It calls method #\_enable.
* @method enable
* @member ns.widget.BaseWidget
* @chainable
*/
prototype.enable = function () {
var self = this,
args = slice.call(arguments),
element = self.element;
element.classList.remove(disableClass);
element.setAttribute(ariaDisabled, false);
if (typeof self._enable === TYPE_FUNCTION) {
args.unshift(element);
self._enable.apply(self, args);
}
return this;
};
/**
* Protected method causing the widget to refresh
* @method _refresh
* @protected
* @member ns.widget.BaseWidget
* @template
*/
/**
* Refreshes widget.
*
* It calls method #\_refresh.
* @method refresh
* @member ns.widget.BaseWidget
* @chainable
*/
prototype.refresh = function () {
var self = this;
if (typeof self._refresh === TYPE_FUNCTION) {
self._refresh.apply(self, arguments);
}
return self;
};
/**
* Gets or sets options of the widget.
*
* This method can work in many context.
*
* If first argument is type of object them, method set values for options given in object. Keys of object are names of options and values from object are values to set.
*
* If you give only one string argument then method return value for given option.
*
* If you give two arguments and first argument will be a string then second argument will be intemperate as value to set.
*
* @method option
* @param {string|Object} [name] name of option
* @param {*} [value] value to set
* @member ns.widget.BaseWidget
* @return {*} return value of option or undefined if method is called in setter context
*/
prototype.option = function (/*name, value*/) {
var self = this,
args = slice.call(arguments),
firstArgument = args.shift(),
secondArgument = args.shift(),
key,
result,
partResult,
refresh = false;
if (typeof firstArgument === "string") {
result = self._oneOption(firstArgument, secondArgument);
if (firstArgument !== undefined && secondArgument !== undefined) {
refresh = result;
result = undefined;
}
} else if (typeof firstArgument === "object") {
for (key in firstArgument) {
if (firstArgument.hasOwnProperty(key)) {
partResult = self._oneOption(key, firstArgument[key]);
if (key !== undefined && firstArgument[key] !== undefined) {
refresh = refresh || partResult;
}
}
}
}
if (refresh) {
self.refresh();
}
return result;
};
/**
* Gets or sets one option of the widget.
*
* @method _oneOption
* @param {string} field
* @param {*} value
* @member ns.widget.BaseWidget
* @return {*}
* @protected
*/
prototype._oneOption = function (field, value) {
var self = this,
methodName,
refresh = false;
if (value === undefined) {
methodName = "_get" + (field[0].toUpperCase() + field.slice(1));
if (typeof self[methodName] === TYPE_FUNCTION) {
return self[methodName]();
}
return self.options[field];
}
methodName = "_set" + (field[0].toUpperCase() + field.slice(1));
if (typeof self[methodName] === TYPE_FUNCTION) {
self[methodName](self.element, value);
} else {
self.options[field] = value;
if (self.element) {
self.element.setAttribute("data-" + (field.replace(/[A-Z]/g, function (c) {
return "-" + c.toLowerCase();
})), value);
refresh = true;
}
}
return refresh;
};
/**
* Returns true if widget has bounded events.
*
* This methods enables to check if the widget has bounded
* events through the {@link ns.widget.BaseWidget#bindEvents} method.
* @method isBound
* @param {string} [type] Type of widget
* @member ns.widget.BaseWidget
* @internal
* @return {boolean} true if events are bounded
*/
prototype.isBound = function (type) {
var element = this.element;
type = type || this.name;
return element && element.hasAttribute(engineDataTau.bound) && element.getAttribute(engineDataTau.bound).indexOf(type) > -1;
};
/**
* Returns true if widget is built.
*
* This methods enables to check if the widget was built
* through the {@link ns.widget.BaseWidget#build} method.
* @method isBuilt
* @param {string} [type] Type of widget
* @member ns.widget.BaseWidget
* @internal
* @return {boolean} true if the widget was built
*/
prototype.isBuilt = function (type) {
var element = this.element;
type = type || this.name;
return element && element.hasAttribute(engineDataTau.built) && element.getAttribute(engineDataTau.built).indexOf(type) > -1;
};
/**
* Protected method getting the value of widget
* @method _getValue
* @return {*}
* @member ns.widget.BaseWidget
* @template
* @protected
*/
/**
* Protected method setting the value of widget
* @method _setValue
* @param {*} value
* @return {*}
* @member ns.widget.BaseWidget
* @template
* @protected
*/
/**
* Gets or sets value of the widget.
*
* @method value
* @param {*} [value] New value of widget
* @member ns.widget.BaseWidget
* @return {*}
*/
prototype.value = function (value) {
var self = this;
if (value !== undefined) {
if (typeof self._setValue === TYPE_FUNCTION) {
return self._setValue(value);
}
return self;
}
if (typeof self._getValue === TYPE_FUNCTION) {
return self._getValue();
}
return self;
};
/**
* Triggers an event on widget's element.
*
* @method trigger
* @param {string} eventName The name of event to trigger
* @param {?*} [data] additional Object to be carried with the event
* @param {boolean} [bubbles=true] Indicating whether the event
* bubbles up through the DOM or not
* @param {boolean} [cancelable=true] Indicating whether
* the event is cancelable
* @member ns.widget.BaseWidget
* @return {boolean} False, if any callback invoked preventDefault on event object
*/
prototype.trigger = function (eventName, data, bubbles, cancelable) {
return eventUtils.trigger(this.element, eventName, data, bubbles, cancelable);
};
/**
* Adds event listener to widget's element.
* @method on
* @param {string} eventName The name of event
* @param {Function} listener Function called after event will be trigger
* @param {boolean} [useCapture=false] useCapture Parameter of addEventListener
* @member ns.widget.BaseWidget
*/
prototype.on = function (eventName, listener, useCapture) {
eventUtils.on(this.element, eventName, listener, useCapture);
};
/**
* Removes event listener from widget's element.
* @method off
* @param {string} eventName The name of event
* @param {Function} listener Function call after event will be trigger
* @param {boolean} [useCapture=false] useCapture Parameter of addEventListener
* @member ns.widget.BaseWidget
*/
prototype.off = function (eventName, listener, useCapture) {
eventUtils.off(this.element, eventName, listener, useCapture);
};
BaseWidget.prototype = prototype;
// definition
ns.widget.BaseWidget = BaseWidget;
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* #Namespace For Widgets
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @class ns.widget
*/
(function (document, ns) {
"use strict";
ns.widget.core = ns.widget.core || {};
}(window.document, ns));
/*global window, define */
/*jslint plusplus: true */
/*jshint -W069 */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @author Jadwiga Sosnowska <j.sosnowska@partner.samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (window, document, ns) {
"use strict";
var DOM = ns.util.DOM,
DASH_TO_UPPER_CASE_REGEXP = /-([a-z])/gi;
/**
* Returns css property for element
* @method getCSSProperty
* @param {HTMLElement} element
* @param {string} property
* @param {string|number|null} [def=null] default returned value
* @param {"integer"|"float"|null} [type=null] auto type casting
* @return {string|number|null}
* @member ns.util.DOM
* @static
*/
function getCSSProperty(element, property, def, type) {
var style = window.getComputedStyle(element),
value = null,
result = def;
if (style) {
value = style.getPropertyValue(property);
if (value) {
switch (type) {
case "integer":
value = parseInt(value, 10);
if (!isNaN(value)) {
result = value;
}
break;
case "float":
value = parseFloat(value);
if (!isNaN(value)) {
result = value;
}
break;
default:
result = value;
break;
}
}
}
return result;
}
/**
* Extracts css properties from computed css for an element.
* The properties values are applied to the specified
* properties list (dictionary)
* @method extractCSSProperties
* @param {HTMLElement} element
* @param {Object} properties
* @param {?string} [pseudoSelector=null]
* @param {boolean} [noConversion=false]
* @member ns.util.DOM
* @static
*/
function extractCSSProperties (element, properties, pseudoSelector, noConversion) {
var style = window.getComputedStyle(element, pseudoSelector),
property,
value = null,
utils = ns.util;
// @TODO extractCSSProperties should rather return raw values (with units)
for (property in properties) {
if (properties.hasOwnProperty(property)) {
value = style.getPropertyValue(property);
if (utils.isNumber(value) && !noConversion) {
if (value.match(/\./gi)) {
properties[property] = parseFloat(value);
} else {
properties[property] = parseInt(value, 10);
}
} else {
properties[property] = value;
}
}
}
}
/**
* Returns elements height from computed style
* @method getElementHeight
* @param {HTMLElement} element
* if null then the "inner" value is assigned
* @param {"outer"|null} [type=null]
* @param {boolean} [includeOffset=false]
* @param {boolean} [includeMargin=false]
* @param {?string} [pseudoSelector=null]
* @param {boolean} [force=false] check even if element is hidden
* @return {number}
* @member ns.util.DOM
* @static
*/
function getElementHeight(element, type, includeOffset, includeMargin, pseudoSelector, force) {
var height = 0,
style,
value,
originalDisplay = null,
originalVisibility = null,
originalPosition = null,
outer = (type && type === "outer") || false,
offsetHeight = 0,
property,
props = {
"height": 0,
"margin-top": 0,
"margin-bottom": 0,
"padding-top": 0,
"padding-bottom": 0,
"border-top-width": 0,
"border-bottom-width": 0,
"box-sizing": ""
};
if (element) {
style = element.style;
if (style.display !== "none") {
extractCSSProperties(element, props, pseudoSelector, true);
offsetHeight = element.offsetHeight;
} else if (force) {
originalDisplay = style.display;
originalVisibility = style.visibility;
originalPosition = style.position;
style.display = "block";
style.visibility = "hidden";
style.position = "relative";
extractCSSProperties(element, props, pseudoSelector, true);
offsetHeight = element.offsetHeight;
style.display = originalDisplay;
style.visibility = originalVisibility;
style.position = originalPosition;
}
// We are extracting raw values to be able to check the units
if(typeof props["height"] === "string" && props["height"].indexOf("px") === -1){
//ignore non px values such as auto or %
props["height"] = 0;
}
for (property in props) {
if (props.hasOwnProperty(property) && property !== "box-sizing"){
value = parseFloat(props[property]);
if (isNaN(value)) {
value = 0;
}
props[property] = value;
}
}
height += props["height"];
if (props["box-sizing"] !== 'border-box') {
height += props["padding-top"] + props["padding-bottom"];
}
if (includeOffset) {
height = offsetHeight;
} else if (outer && props["box-sizing"] !== 'border-box') {
height += props["border-top-width"] + props["border-bottom-width"];
}
if (includeMargin) {
height += Math.max(0, props["margin-top"]) + Math.max(0, props["margin-bottom"]);
}
}
return height;
}
/**
* Returns elements width from computed style
* @method getElementWidth
* @param {HTMLElement} element
* if null then the "inner" value is assigned
* @param {"outer"|null} [type=null]
* @param {boolean} [includeOffset=false]
* @param {boolean} [includeMargin=false]
* @param {?string} [pseudoSelector=null]
* @param {boolean} [force=false] check even if element is hidden
* @return {number}
* @member ns.util.DOM
* @static
*/
function getElementWidth(element, type, includeOffset, includeMargin, pseudoSelector, force) {
var width = 0,
style,
value,
originalDisplay = null,
originalVisibility = null,
originalPosition = null,
offsetWidth = 0,
property,
outer = (type && type === "outer") || false,
props = {
"width": 0,
"margin-left": 0,
"margin-right": 0,
"padding-left": 0,
"padding-right": 0,
"border-left-width": 0,
"border-right-width": 0,
"box-sizing": ""
};
if (element) {
style = element.style;
if (style.display !== "none") {
extractCSSProperties(element, props, pseudoSelector, true);
offsetWidth = element.offsetWidth;
} else if (force) {
originalDisplay = style.display;
originalVisibility = style.visibility;
originalPosition = style.position;
style.display = "block";
style.visibility = "hidden";
style.position = "relative";
extractCSSProperties(element, props, pseudoSelector, true);
style.display = originalDisplay;
style.visibility = originalVisibility;
style.position = originalPosition;
}
if(typeof props["width"] === 'string' && props["width"].indexOf("px") === -1) {
//ignore non px values such as auto or %
props["width"] = 0;
}
for (property in props) {
if (props.hasOwnProperty(property) && property !== "box-sizing"){
value = parseFloat(props[property]);
if (isNaN(value)) {
value = 0;
}
props[property] = value;
}
}
width += props["width"];
if (props["box-sizing"] !== 'border-box') {
width += props["padding-left"] + props["padding-right"];
}
if (includeOffset) {
width = offsetWidth;
} else if (outer && props["box-sizing"] !== 'border-box') {
width += props["border-left-width"] + props["border-right-width"];
}
if (includeMargin) {
width += Math.max(0, props["margin-left"]) + Math.max(0, props["margin-right"]);
}
}
return width;
}
/**
* Returns offset of element
* @method getElementOffset
* @param {HTMLElement} element
* @return {Object}
* @member ns.util.DOM
* @static
*/
function getElementOffset(element) {
var left = 0,
top = 0;
do {
top += element.offsetTop;
left += element.offsetLeft;
element = element.offsetParent;
} while (element !== null);
return {
top: top,
left: left
};
}
/**
* Check if element occupies place at view
* @method isOccupiedPlace
* @param {HTMLElement} element
* @return {boolean}
* @member ns.util.DOM
* @static
*/
function isOccupiedPlace(element) {
return !(element.offsetWidth <= 0 && element.offsetHeight <= 0);
}
function toUpperCaseFn(match, value) {
return value.toLocaleUpperCase();
}
function dashesToCamelCase(str) {
return str.replace(DASH_TO_UPPER_CASE_REGEXP, toUpperCaseFn);
}
function firstToUpperCase(str) {
return str.charAt(0).toLocaleUpperCase() + str.substring(1);
}
/**
* Set values for element with prefixes for browsers
* @method setPrefixedStyle
* @param {HTMLElement} element
* @param {string} property
* @param {string|Object|null} value
* @member ns.util.DOM
* @static
*/
function setPrefixedStyle(element, property, value) {
var style = element.style,
propertyForPrefix = firstToUpperCase(dashesToCamelCase(property)),
values = (typeof value === "string") ? {
webkit: value,
moz: value,
o: value,
ms: value,
normal: value
} : value;
style[property] = values.normal;
style["webkit" + propertyForPrefix] = values.webkit;
style["moz" + propertyForPrefix] = values.moz;
style["o" + propertyForPrefix] = values.o;
style["ms" + propertyForPrefix] = values.ms;
}
/**
* Get value from element with prefixes for browsers
* @method getCSSProperty
* @param {string} value
* @return {Object}
* @member ns.util.DOM
* @static
*/
function getPrefixedValue(value) {
return {
webkit: "-webkit-" + value,
moz: "-moz-" + value,
o: "-ms-" + value,
ms: "-o-" + value,
normal: value
};
}
/**
* Returns style value for css property with browsers prefixes
* @method getPrefixedStyle
* @param {HTMLStyle} styles
* @param {string} property
* @return {Object}
* @member ns.util.DOM
* @static
*/
function getPrefixedStyleValue(styles, property) {
var prefixedProperties = getPrefixedValue(property),
value,
key;
for (key in prefixedProperties) {
value = styles[prefixedProperties[key]];
if (value && value !== "none") {
return value;
}
}
return value;
}
// assign methods to namespace
DOM.getCSSProperty = getCSSProperty;
DOM.extractCSSProperties = extractCSSProperties;
DOM.getElementHeight = getElementHeight;
DOM.getElementWidth = getElementWidth;
DOM.getElementOffset = getElementOffset;
DOM.isOccupiedPlace = isOccupiedPlace;
DOM.setPrefixedStyle = setPrefixedStyle;
DOM.getPrefixedValue = getPrefixedValue;
DOM.getPrefixedStyleValue = getPrefixedStyleValue;
}(window, window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Page Widget
* Page is main element of application's structure.
*
* ## Default selectors
* In the Tizen Web UI framework the application page structure is based on a header, content and footer elements:
*
* - **The header** is placed at the top, and displays the page title and optionally buttons.
* - **The content** is the section below the header, showing the main content of the page.
* - **The footer** is a bottom part of page which can display for example buttons
*
* The following table describes the specific information for each section.
*
* <table>
* <tr>
* <th>Section</th>
* <th>Class</th>
* <th>Mandatory</th>
* <th>Description</th>
* </tr>
* <tr>
* <td rowspan="2">Page</td>
* <td>ui-page</td>
* <td>Yes</td>
* <td>Defines the element as a page.
*
* The page widget is used to manage a single item in a page-based architecture.
*
* A page is composed of header (optional), content (mandatory), and footer (optional) elements.</td>
* </tr>
* <tr>
* <td>ui-page-active</td>
* <td>No</td>
* <td>If an application has a static start page, insert the ui-page-active class in the page element to speed up the application launch. The start page with the ui-page-active class can be displayed before the framework is fully loaded.
*
*If this class is not used, the framework inserts the class automatically to the first page of the application. However, this has a slowing effect on the application launch, because the page is displayed only after the framework is fully loaded.</td>
* </tr>
* <tr>
* <td>Header</td>
* <td>ui-header</td>
* <td>No</td>
* <td>Defines the element as a header.</td>
* </tr>
* <tr>
* <td>Content</td>
* <td>ui-content</td>
* <td>Yes</td>
* <td>Defines the element as content.</td>
* </tr>
* <tr>
* <td>Footer</td>
* <td>ui-footer</td>
* <td>No</td>
* <td>Defines the element as a footer.
*
* The footer section is mostly used to include option buttons.</td>
* </tr>
* </table>
*
* All elements with class=ui-page will be become page widgets
*
* @example
* <!--Page layout-->
* <div class="ui-page ui-page-active">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <footer class="ui-footer"></footer>
* </div>
*
* <!--Page layout with more button in header-->
* <div class="ui-page ui-page-active">
* <header class="ui-header ui-has-more">
* <h2 class="ui-title">Call menu</h2>
* <button type="button" class="ui-more ui-icon-overflow">More Options</button>
* </header>
* <div class="ui-content">Content message</div>
* <footer class="ui-footer">
* <button type="button" class="ui-btn">Footer Button</button>
* </footer>
* </div>
*
* ## Manual constructor
* For manual creation of page widget you can use constructor of widget from **tau** namespace:
*
* @example
* var pageElement = document.getElementById("page"),
* page = tau.widget.page(buttonElement);
*
* Constructor has one require parameter **element** which are base **HTMLElement** to create widget. We recommend get this element by method *document.getElementById*.
*
* ## Multi-page Layout
*
* You can implement a template containing multiple page containers in the application index.html file.
*
* In the multi-page layout, the main page is defined with the ui-page-active class. If no page has the ui-page-active class, the framework automatically sets up the first page in the source order as the main page. You can improve the launch performance by explicitly defining the main page to be displayed first. If the application has to wait for the framework to set up the main page, the page is displayed with some delay only after the framework is fully loaded.
*
* You can link to internal pages by referring to the ID of the page. For example, to link to the page with an ID of two, the link element needs the href="#two" attribute in the code, as in the following example.
*
* @example
* <!--Main page-->
* <div id="one" class="ui-page ui-page-active">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <footer class="ui-footer"></footer>
* </div>
*
* <!--Secondary page-->
* <div id="two" class="ui-page">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <footer class="ui-footer"></footer>
* </div>
*
* To find the currently active page, use the ui-page-active class.
*
* ## Changing Pages
* ### Go to page in JavaScript
* To change page use method *tau.changePage*
*
* @example
* tau.changePage("page-two");
*
* ### Back in JavaScript
* To back to previous page use method *tau.back*
*
* @example
* tau.back();
*
* ## Transitions
*
* When changing the active page, you can use a page transition.
*
* Tizen Web UI Framework does not apply transitions by default. To set a custom transition effect, you must add the data-transition attribute to a link:
*
* @example
* <a href="index.html" data-transition="slideup">I'll slide up</a>
*
* To set a default custom transition effect for all pages, use the pageTransition property:
*
* @example
* tau.defaults.pageTransition = "slideup";
*
* ### Transitions list
*
* - **none** no transition.
* - **slideup** Makes the content of the next page slide up, appearing to conceal the content of the previous page.
*
* ## Handling Page Events
*
* With page widget we have connected many of events.
*
* To handle page events, use the following code:
*
* @example
* <div id="page" class="ui-page">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* </div>
*
* <script>
* var page = document.getElementById("page");
* page.addEventListener("Event", function(event) {
* // Your code
* });
* </script>
*
* To bind an event callback on the Back key, use the following code:
*
* Full list of available events is in [events list section](#events-list).
*
* To bind an event callback on the Back key, use the following code:
*
* @example
* <script>
* window.addEventListener("tizenhwkey", function (event) {
* if (event.keyName == "back") {
* // Call window.history.back() to go to previous browser window
* // Call tizen.application.getCurrentApplication().exit() to exit application
* // Add script to add another behavior
* }
* });
* </script>
*
* ## Options for Page Widget
*
* Page widget hasn't any options.
*
* ## Methods
*
* To call method on widget you can use tau API:
*
* @example
* var pageElement = document.getElementById("page"),
* page = tau.widget.page(buttonElement);
*
* page.methodName(methodArgument1, methodArgument2, ...);
*
* @class ns.widget.core.Page
* @extends ns.widget.BaseWidget
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
* @author Damian Osipiuk <d.osipiuk@samsung.com>
*/
(function (document, ns) {
"use strict";
/**
* Alias for {@link ns.widget.BaseWidget}
* @property {Object} BaseWidget
* @member ns.widget.core.Page
* @private
* @static
*/
var BaseWidget = ns.widget.BaseWidget,
/**
* Alias for {@link ns.util}
* @property {Object} util
* @member ns.widget.core.Page
* @private
* @static
*/
util = ns.util,
/**
* Alias for {@link ns.util.DOM}
* @property {Object} doms
* @member ns.widget.core.Page
* @private
* @static
*/
doms = util.DOM,
/**
* Alias for {@link ns.util.selectors}
* @property {Object} utilSelectors
* @member ns.widget.core.Page
* @private
* @static
*/
utilSelectors = util.selectors,
/**
* Alias for {@link ns.engine}
* @property {Object} engine
* @member ns.widget.core.Page
* @private
* @static
*/
engine = ns.engine,
Page = function () {
var self = this;
/**
* Callback on resize
* @property {?Function} contentFillAfterResizeCallback
* @private
* @member ns.widget.core.Page
*/
self.contentFillAfterResizeCallback = null;
self._initialContentStyle = {};
/**
* Options for widget.
* It is empty object, because widget Page does not have any options.
* @property {Object} options
* @member ns.widget.core.Page
*/
self.options = {};
self._contentStyleAttributes = ["height", "width", "minHeight", "marginTop", "marginBottom"];
self._ui = {};
},
/**
* Dictionary for page related event types
* @property {Object} EventType
* @member ns.widget.core.Page
* @static
*/
EventType = {
/**
* Triggered on the page we are transitioning to,
* after the transition animation has completed.
* @event pageshow
* @member ns.widget.core.Page
*/
SHOW: "pageshow",
/**
* Triggered on the page we are transitioning away from,
* after the transition animation has completed.
* @event pagehide
* @member ns.widget.core.Page
*/
HIDE: "pagehide",
/**
* Triggered when the page has been created in the DOM
* (for example, through Ajax) but before all widgets
* have had an opportunity to enhance the contained markup.
* @event pagecreate
* @member ns.widget.core.Page
*/
CREATE: "pagecreate",
/**
* Triggered when the page is being initialized,
* before most plugin auto-initialization occurs.
* @event pagebeforecreate
* @member ns.widget.core.Page
*/
BEFORE_CREATE: "pagebeforecreate",
/**
* Triggered on the page we are transitioning to,
* before the actual transition animation is kicked off.
* @event pagebeforeshow
* @member ns.widget.core.Page
*/
BEFORE_SHOW: "pagebeforeshow",
/**
* Triggered on the page we are transitioning away from,
* before the actual transition animation is kicked off.
* @event pagebeforehide
* @member ns.widget.core.Page
*/
BEFORE_HIDE: "pagebeforehide"
},
/**
* Dictionary for page related css class names
* @property {Object} classes
* @member ns.widget.core.Page
* @static
* @readonly
*/
classes = {
uiPage: "ui-page",
uiPageActive: "ui-page-active",
uiSection: "ui-section",
uiHeader: "ui-header",
uiFooter: "ui-footer",
uiContent: "ui-content"
},
prototype = new BaseWidget();
Page.classes = classes;
Page.events = EventType;
/**
* Configure default options for widget
* @method _configure
* @protected
* @member ns.widget.core.Page
*/
prototype._configure = function() {
var options = this.options || {};
/**
* Object with default options
* @property {Object} options
* @property {boolean|string|null} [options.header=false] Sets content of header.
* @property {boolean|string|null} [options.footer=false] Sets content of footer.
* @property {string} [options.content=null] Sets content of popup.
* @member ns.widget.core.Page
* @static
*/
options.header = null;
options.footer = null;
options.content = null;
this.options = options;
};
/**
* Sets top-bottom css attributes for content element
* to allow it to fill the page dynamically
* @method _contentFill
* @member ns.widget.core.Page
*/
prototype._contentFill = function () {
var self = this,
element = self.element,
screenWidth = window.innerWidth,
screenHeight = window.innerHeight,
contentSelector = classes.uiContent,
headerSelector = classes.uiHeader,
footerSelector = classes.uiFooter,
extraHeight = 0,
children = [].slice.call(element.children),
childrenLength = children.length,
elementStyle = element.style,
i,
node,
contentStyle,
marginTop,
marginBottom,
nodeStyle;
elementStyle.width = screenWidth + "px";
elementStyle.height = screenHeight + "px";
for (i = 0; i < childrenLength; i++) {
node = children[i];
if (node.classList.contains(headerSelector) ||
node.classList.contains(footerSelector)) {
extraHeight += doms.getElementHeight(node);
}
}
for (i = 0; i < childrenLength; i++) {
node = children[i];
nodeStyle = node.style;
if (node.classList.contains(contentSelector)) {
contentStyle = window.getComputedStyle(node);
marginTop = parseFloat(contentStyle.marginTop);
marginBottom = parseFloat(contentStyle.marginBottom);
nodeStyle.height = (screenHeight - extraHeight - marginTop - marginBottom) + "px";
nodeStyle.width = screenWidth + "px";
}
}
};
prototype._storeContentStyle = function () {
var initialContentStyle = this._initialContentStyle,
contentStyleAttributes = this._contentStyleAttributes,
content = this.element.querySelector("." + classes.uiContent),
contentStyle = content ? content.style : {};
contentStyleAttributes.forEach(function(name) {
initialContentStyle[name] = contentStyle[name];
});
};
prototype._restoreContentStyle = function () {
var initialContentStyle = this._initialContentStyle,
contentStyleAttributes = this._contentStyleAttributes,
content = this.element.querySelector("." + classes.uiContent),
contentStyle = content ? content.style : {};
contentStyleAttributes.forEach(function(name) {
contentStyle[name] = initialContentStyle[name];
});
};
/**
* Setter for footer option
* @method _setFooter
* @param {HTMLElement} element
* @param {string} value
* @protected
* @member ns.widget.core.Page
*/
prototype._setFooter = function(element, value) {
var self = this,
ui = self._ui,
footer = ui.footer;
// footer element if footer does not exist and value is true or string
if (!footer && value) {
footer = document.createElement("footer");
element.appendChild(footer);
ui.footer = footer;
}
if (footer) {
// remove child if footer does not exist and value is set to false
if (value === false) {
element.removeChild(footer);
} else {
// if options is set to true, to string or not is set
// add class
footer.classList.add(classes.uiFooter);
// if is string fill content by string value
if (typeof value === "string") {
ui.footer.textContent = value;
}
}
// and remember options
self.options.footer = value;
}
};
/**
* Setter for header option
* @method _setHeader
* @param {HTMLElement} element
* @param {string} value
* @protected
* @member ns.widget.core.Page
*/
prototype._setHeader = function(element, value) {
var self = this,
ui = self._ui,
header = ui.header;
// header element if header does not exist and value is true or string
if (!header && value) {
header = document.createElement("header");
element.appendChild(header);
ui.header = header;
}
if (header) {
// remove child if header does not exist and value is set to false
if (value === false) {
element.removeChild(header);
} else {
// if options is set to true, to string or not is set
// add class
header.classList.add(classes.uiHeader);
// if is string fill content by string value
if (typeof value === "string") {
ui.header.textContent = value;
}
}
// and remember options
self.options.header = value;
}
};
/**
* Setter for content option
* @method _setContent
* @param {HTMLElement} element
* @param {string} value
* @protected
* @member ns.widget.core.Page
*/
prototype._setContent = function(element, value) {
var self = this,
ui = self._ui,
content = ui.content,
child = element.firstChild,
next;
if (!content && value) {
content = document.createElement("div");
while (child) {
next = child.nextSibling;
if (child !== ui.footer && child !== ui.header) {
content.appendChild(child);
}
child = next;
}
element.insertBefore(content, ui.footer);
ui.content = content;
}
if (content) {
// remove child if content exist and value is set to false
if (value === false) {
element.removeChild(content);
} else {
// if options is set to true, to string or not is set
// add class
content.classList.add(classes.uiContent);
// if is string fill content by string value
if (typeof value === "string") {
content.textContent = value;
}
}
// and remember options
self.options.content = value;
}
};
/**
* Method creates empty page header. It also checks for additional
* content to be added in header.
* @method _buildHeader
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Page
*/
prototype._buildHeader = function(element) {
var self = this;
self._ui.header = utilSelectors.getChildrenBySelector(element, "header,[data-role='header'],." + classes.uiHeader)[0];
self._setHeader(element, self.options.header);
};
/**
* Method creates empty page footer.
* @method _buildFooter
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Page
*/
prototype._buildFooter = function(element) {
var self = this;
self._ui.footer = utilSelectors.getChildrenBySelector(element, "footer,[data-role='footer'],." + classes.uiFooter)[0];
self._setFooter(element, self.options.footer);
};
/**
* Method creates empty page content.
* @method _buildContent
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Page
*/
prototype._buildContent = function(element) {
var self = this;
self._ui.content = utilSelectors.getChildrenBySelector(element, "[data-role='content'],." + classes.uiContent)[0];
self._setContent(element, self.options.content);
};
/**
* Build page
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.core.Page
*/
prototype._build = function (element) {
var self = this;
element.classList.add(classes.uiPage);
self._buildHeader(element);
self._buildFooter(element);
self._buildContent(element);
return element;
};
/**
* This method sets page active or inactive.
* @method setActive
* @param {boolean} value If true, then page will be active. Otherwise, page will be inactive.
* @member ns.widget.core.Page
*/
prototype.setActive = function (value) {
var elementClassList = this.element.classList;
if (value) {
this.focus();
elementClassList.add(classes.uiPageActive);
} else {
this.blur();
elementClassList.remove(classes.uiPageActive);
}
};
/**
* Return current status of page.
* @method isActive
* @member ns.widget.core.Page
* @instance
*/
prototype.isActive = function () {
return this.element.classList.contains(classes.uiPageActive);
};
/**
* Sets the focus to page
* @method focus
* @member ns.widget.core.Page
*/
prototype.focus = function () {
var element = this.element,
focusable = element.querySelector("[autofocus]") || element;
focusable.focus();
};
/**
* Removes focus from page and all descendants
* @method blur
* @member ns.widget.core.Page
*/
prototype.blur = function () {
var element = this.element,
focusable = element.querySelector(":focus") || element;
focusable.blur();
};
/**
* Bind events to widget
* @method _bindEvents
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Page
*/
prototype._bindEvents = function (element) {
var self = this;
self.contentFillAfterResizeCallback = self._contentFill.bind(self);
window.addEventListener("resize", self.contentFillAfterResizeCallback, false);
};
/**
* Refresh widget structure
* @method _refresh
* @protected
* @member ns.widget.core.Page
*/
prototype._refresh = function () {
this._restoreContentStyle();
this._contentFill();
};
/**
* Layouting page structure
* @method layout
* @member ns.widget.core.Page
*/
prototype.layout = function () {
this._storeContentStyle();
this._contentFill();
};
/**
* This method triggers BEFORE_SHOW event.
* @method onBeforeShow
* @member ns.widget.core.Page
*/
prototype.onBeforeShow = function () {
this.trigger(EventType.BEFORE_SHOW);
};
/**
* This method triggers SHOW event.
* @method onShow
* @member ns.widget.core.Page
*/
prototype.onShow = function () {
this.trigger(EventType.SHOW);
};
/**
* This method triggers BEFORE_HIDE event.
* @method onBeforeHide
* @member ns.widget.core.Page
*/
prototype.onBeforeHide = function () {
this.trigger(EventType.BEFORE_HIDE);
};
/**
* This method triggers HIDE event.
* @method onHide
* @member ns.widget.core.Page
*/
prototype.onHide = function () {
this._restoreContentStyle();
this.trigger(EventType.HIDE);
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.core.Page
*/
prototype._destroy = function () {
var self = this,
element = self.element;
element = element || self.element;
window.removeEventListener("resize", self.contentFillAfterResizeCallback, false);
// destroy widgets on children
engine.destroyAllWidgets(element, true);
};
Page.prototype = prototype;
Page.createEmptyElement = function() {
var div = document.createElement("div");
div.classList.add(classes.uiPage);
doms.setNSData(div, "role", "page");
return div;
};
// definition
ns.widget.core.Page = Page;
engine.defineWidget(
"Page",
"[data-role=page],.ui-page",
[
"layout",
"focus",
"blur",
"setActive",
"isActive"
],
Page,
"core"
);
engine.defineWidget(
"page",
"",
[
"layout",
"focus",
"blur",
"setActive",
"isActive"
],
Page,
"core"
);
// @remove
// THIS IS ONLY FOR COMPATIBILITY
ns.widget.page = ns.widget.Page;
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # PageContainer Widget
* PageContainer is a widget, which is supposed to have multiple child pages but display only one at a time.
*
* It allows for adding new pages, switching between them and displaying progress bars indicating loading process.
*
* @class ns.widget.core.PageContainer
* @extends ns.widget.BaseWidget
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
* @author Krzysztof Głodowski <k.glodowski@samsung.com>
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
Page = ns.widget.core.Page,
util = ns.util,
eventUtils = ns.event,
DOM = util.DOM,
engine = ns.engine,
classes = {
pageContainer: "ui-page-container",
uiViewportTransitioning: "ui-viewport-transitioning",
out: "out",
in: "in",
reverse: "reverse",
uiPreIn: "ui-pre-in",
uiBuild: "ui-page-build"
},
PageContainer = function () {
/**
* Active page.
* @property {ns.widget.core.Page} [activePage]
* @member ns.widget.core.PageContainer
*/
this.activePage = null;
this.inTransition = false;
},
EventType = {
/**
* Triggered before the changePage() request
* has started loading the page into the DOM.
* @event pagebeforechange
* @member ns.widget.core.PageContainer
*/
PAGE_BEFORE_CHANGE: "pagebeforechange",
/**
* Triggered after the changePage() request
* has finished loading the page into the DOM and
* all page transition animations have completed.
* @event pagechange
* @member ns.widget.core.PageContainer
*/
PAGE_CHANGE: "pagechange",
PAGE_REMOVE: "pageremove"
},
animationend = "animationend",
webkitAnimationEnd = "webkitAnimationEnd",
mozAnimationEnd = "mozAnimationEnd",
msAnimationEnd = "msAnimationEnd",
oAnimationEnd = "oAnimationEnd",
prototype = new BaseWidget();
/**
* Dictionary for PageContainer related event types.
* @property {Object} events
* @property {string} [events.PAGE_CHANGE="pagechange"]
* @member ns.router.route.popup
* @static
*/
PageContainer.events = EventType;
/**
* Dictionary for PageContainer related css class names
* @property {Object} classes
* @member ns.widget.core.Page
* @static
* @readonly
*/
PageContainer.classes = classes;
/**
* This method changes active page to specified element.
* @method change
* @param {HTMLElement} toPage The element to set
* @param {Object} [options] Additional options for the transition
* @param {string} [options.transition=none] Specifies the type of transition
* @param {boolean} [options.reverse=false] Specifies the direction of transition
* @member ns.widget.core.PageContainer
*/
prototype.change = function (toPage, options) {
var self = this,
fromPageWidget = self.getActivePage(),
toPageWidget;
options = options || {};
options.widget = options.widget || "Page";
// The change should be made only if no active page exists
// or active page is changed to another one.
if (!fromPageWidget || (fromPageWidget.element !== toPage)) {
if (toPage.parentNode !== self.element) {
toPage = self._include(toPage);
}
self.trigger(EventType.PAGE_BEFORE_CHANGE);
toPage.classList.add(classes.uiBuild);
toPageWidget = engine.instanceWidget(toPage, options.widget);
// set sizes of page for correct display
toPageWidget.layout();
if (ns.getConfig("autoBuildOnPageChange", false)) {
engine.createWidgets(toPage);
}
if (fromPageWidget) {
fromPageWidget.onBeforeHide();
}
toPageWidget.onBeforeShow();
toPage.classList.remove(classes.uiBuild);
options.deferred = {
resolve: function () {
if (fromPageWidget) {
fromPageWidget.onHide();
if (options.reverse) {
fromPageWidget.destroy();
}
self._removeExternalPage(fromPageWidget, options);
}
toPageWidget.onShow();
self.trigger(EventType.PAGE_CHANGE);
}
};
self._transition(toPageWidget, fromPageWidget, options);
}
};
/**
* This method performs transition between the old and a new page.
* @method _transition
* @param {ns.widget.core.Page} toPageWidget The new page
* @param {ns.widget.core.Page} fromPageWidget The page to be replaced
* @param {Object} [options] Additional options for the transition
* @param {string} [options.transition=none] The type of transition
* @param {boolean} [options.reverse=false] Specifies transition direction
* @param {Object} [options.deferred] Deferred object
* @member ns.widget.core.PageContainer
* @protected
*/
prototype._transition = function (toPageWidget, fromPageWidget, options) {
var self = this,
element = self.element,
elementClassList = element.classList,
transition = !fromPageWidget || !options.transition ? "none" : options.transition,
deferred = options.deferred,
clearClasses = [classes.in, classes.out, classes.uiPreIn, transition],
oldDeferredResolve,
classlist,
oneEvent;
if (options.reverse) {
clearClasses.push(classes.reverse);
}
self.inTransition = true;
elementClassList.add(classes.uiViewportTransitioning);
oldDeferredResolve = deferred.resolve;
deferred.resolve = function () {
var fromPageWidgetClassList = fromPageWidget && fromPageWidget.element.classList,
toPageWidgetClassList = toPageWidget.element.classList;
self._setActivePage(toPageWidget);
elementClassList.remove(classes.uiViewportTransitioning);
self.inTransition = false;
clearClasses.forEach(function (className) {
toPageWidgetClassList.remove(className);
});
if (fromPageWidgetClassList) {
clearClasses.forEach(function (className) {
fromPageWidgetClassList.remove(className);
});
}
oldDeferredResolve();
};
if (transition !== "none") {
oneEvent = function () {
eventUtils.off(
toPageWidget.element,
[
animationend,
webkitAnimationEnd,
mozAnimationEnd,
msAnimationEnd,
oAnimationEnd
],
oneEvent,
false
);
deferred.resolve();
};
eventUtils.on(
toPageWidget.element,
[
animationend,
webkitAnimationEnd,
mozAnimationEnd,
msAnimationEnd,
oAnimationEnd
],
oneEvent,
false
);
if (fromPageWidget) {
classlist = fromPageWidget.element.classList;
classlist.add(transition);
classlist.add(classes.out);
if (options.reverse) {
classlist.add(classes.reverse);
}
}
classlist = toPageWidget.element.classList;
classlist.add(transition);
classlist.add(classes.in);
classlist.add(classes.uiPreIn);
if (options.reverse) {
classlist.add(classes.reverse);
}
} else {
window.setTimeout(deferred.resolve, 0);
}
};
/**
* This method adds an element as a page.
* @method _include
* @param {HTMLElement} page an element to add
* @return {HTMLElement}
* @member ns.widget.core.PageContainer
* @protected
*/
prototype._include = function (page) {
var element = this.element;
if (page.parentNode !== element) {
page = util.importEvaluateAndAppendElement(page, element);
}
return page;
};
/**
* This method sets currently active page.
* @method _setActivePage
* @param {ns.widget.core.Page} page a widget to set as the active page
* @member ns.widget.core.PageContainer
* @protected
*/
prototype._setActivePage = function (page) {
var self = this;
if (self.activePage) {
self.activePage.setActive(false);
}
self.activePage = page;
page.setActive(true);
};
/**
* This method returns active page widget.
* @method getActivePage
* @member ns.widget.core.PageContainer
* @return {ns.widget.core.Page} Currently active page
*/
prototype.getActivePage = function () {
return this.activePage;
};
/**
* This method displays a progress bar indicating loading process.
* @method showLoading
* @member ns.widget.core.PageContainer
* @return {null}
*/
prototype.showLoading = function () {
return null;
};
/**
* This method hides any active progress bar.
* @method hideLoading
* @member ns.widget.core.PageContainer
* @return {null}
*/
prototype.hideLoading = function () {
return null;
};
/**
* This method removes page element from the given widget and destroys it.
* @method _removeExternalPage
* @param {ns.widget.core.Page} fromPageWidget the widget to destroy
* @param {Object} [options] transition options
* @param {boolean} [options.reverse=false] specifies transition direction
* @member ns.widget.core.PageContainer
* @protected
*/
prototype._removeExternalPage = function ( fromPageWidget, options) {
var fromPage = fromPageWidget.element;
options = options || {};
if (options.reverse && DOM.hasNSData(fromPage, "external")) {
if (fromPage.parentNode) {
fromPage.parentNode.removeChild(fromPage);
this.trigger(EventType.PAGE_REMOVE);
}
}
};
PageContainer.prototype = prototype;
// definition
ns.widget.core.PageContainer = PageContainer;
engine.defineWidget(
"pagecontainer",
"",
["change", "getActivePage", "showLoading", "hideLoading"],
PageContainer,
"core"
);
}(window.document, ns));
/*global window, define, ns */
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
Button = function () {
var self = this;
self.options = {};
},
classes = {
BTN: "ui-btn",
DISABLED: "ui-state-disabled",
INLINE: "ui-inline",
BTN_ICON: "ui-btn-icon",
ICON_PREFIX: "ui-icon-",
BTN_CIRCLE: "ui-btn-circle",
BTN_NOBG: "ui-btn-nobg",
BTN_ICON_ONLY: "ui-btn-icon-only",
BTN_ICON_POSITION_PREFIX: "ui-btn-icon-"
},
buttonStyle = {
CIRCLE: "circle",
NOBG: "nobg"
},
prototype = new BaseWidget();
Button.classes = classes;
Button.prototype = prototype;
/**
* Configure button
* @method _configre
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Button
*/
prototype._configure = function () {
this.options = {
// common options
inline: false,//url
icon: null,
disabled: false,
// mobile options
style: null,
iconpos: "left"
};
};
/**
* Set style option
* @method _setStyle
* @param {HTMLElement} element
* @param {string} style
* @protected
* @member ns.widget.core.Button
*/
prototype._setStyle = function (element, style) {
var options = this.options,
buttonClassList = element.classList;
style = style || options.style;
switch (style) {
case buttonStyle.CIRCLE:
buttonClassList.remove(classes.BTN_NOBG);
buttonClassList.add(classes.BTN_CIRCLE);
break;
case buttonStyle.NOBG:
buttonClassList.remove(classes.BTN_CIRCLE);
buttonClassList.add(classes.BTN_NOBG);
break;
default:
}
};
/**
* Set inline option
* @method _setInline
* @param {HTMLElement} element
* @param {boolean} inline
* @protected
* @member ns.widget.core.Button
*/
prototype._setInline = function (element, inline) {
var options = this.options;
inline = inline || options.inline;
if (inline) {
element.classList.add(classes.INLINE);
}
};
/**
* Set icon option
* @method _setIcon
* @param {HTMLElement} element
* @param {string} icon
* @protected
* @member ns.widget.core.Button
*/
prototype._setIcon = function (element, icon) {
var self = this,
options = self.options;
icon = icon || options.icon;
if (icon) {
element.classList.add(classes.BTN_ICON);
element.classList.add(classes.ICON_PREFIX + icon);
self._setTitleForIcon(element);
}
};
/**
* Set iconpos option
* @method _setIconpos
* @param {HTMLElement} element
* @param {string} iconpos
* @protected
* @member ns.widget.core.Button
*/
prototype._setIconpos = function (element, iconpos) {
var options = this.options,
style = options.style,
innerTextLength = element.textContent.length || (element.value ? element.value.length : 0);
iconpos = iconpos || options.iconpos;
if (options.icon && style !== buttonStyle.CIRCLE && style !== buttonStyle.NOBG) {
if (innerTextLength > 0) {
element.classList.add(classes.BTN_ICON_POSITION_PREFIX + iconpos);
} else {
element.classList.add(classes.BTN_ICON_ONLY);
}
}
};
/**
* Set title for button without showing text
* @method _setTitleForIcon
* @param {HTMLElement|HTMLInputElement|HTMLButtonElement} element
* @protected
* @member ns.widget.core.Button
*/
prototype._setTitleForIcon = function (element) {
var options = this.options,
buttonText = element.textContent;
// Add title to element if button not has text.
if (options.iconpos === "notext" && !element.getAttribute("title")) {
element.setAttribute("title", buttonText);
ns.warn("iconpos='notext' is deprecated.");
}
};
/**
* Sets button to disabled if element.disabled or element.disabled property is true,
* or class is set to ui-state-disabled
* @method _setDisabled
* @param {HTMLElement} element
* @protected
*/
prototype._setDisabled = function (element) {
var self = this,
options = self.options,
buttonClassList = element.classList;
if (options.disabled === true || element.disabled || buttonClassList.contains(classes.DISABLED)) {
self._disable(element);
}
};
/**
* Build Button
* @method _build
* @protected
* @param {HTMLElement} element
* @return {HTMLElement}
* @member ns.widget.core.Button
*/
prototype._build = function (element) {
var self = this,
buttonClassList = element.classList;
if (!buttonClassList.contains(classes.BTN)) {
buttonClassList.add(classes.BTN);
}
self._setStyle(element);
self._setInline(element);
self._setIconpos(element);
self._setIcon(element);
self._setDisabled(element);
return element;
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.core.Button
*/
prototype._refresh = function () {
var self = this,
element = this.element;
self._setStyle(element);
self._setInline(element);
self._setIconpos(element);
self._setIcon(element);
self._setDisabled(element);
return null;
};
/**
* Get value of button
* @method _getValue
* @protected
* @member ns.widget.core.Button
*/
prototype._getValue = function () {
return this.element.textContent;
};
/**
* Set value of button
* @method _setValue
* @param {string} value
* @protected
* @member ns.widget.core.Button
*/
prototype._setValue = function (value) {
this.element.textContent = value;
};
/**
* Enable button
* @method _enable
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Button
*/
prototype._enable = function (element) {
var options = this.options;
if (element) {
if (element.tagName.toLowerCase() === "button") {
element.removeAttribute("disabled");
}
element.classList.remove(classes.DISABLED);
options.disabled = false;
}
};
/**
* Disable button
* @method _disable
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Button
*/
prototype._disable = function (element) {
var options = this.options;
if (element) {
if (element.tagName.toLowerCase() === "button") {
element.disabled = true;
}
element.classList.add(classes.DISABLED);
options.disabled = true;
}
};
ns.widget.core.Button = Button;
engine.defineWidget(
"Button",
"button, [data-role='button'], .ui-btn, input[type='button']",
[],
Button,
"core"
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
Checkbox = function () {
var self = this;
self._inputtype = null;
},
classes = {
checkbox: "ui-checkbox"
},
prototype = new BaseWidget();
Checkbox.prototype = prototype;
/**
* Build Checkbox widget
* @method _build
* @param {HTMLElement} element
* @protected
* @member ns.widget.Checkbox
* @instance
*/
prototype._build = function (element) {
var inputType = element.getAttribute("type");
if (inputType !== "checkbox") {
//_build should always return element
return element;
}
element.classList.add(classes.checkbox);
return element;
};
/**
* Returns the value of checkbox
* @method _getValue
* @member ns.widget.Checkbox
* @return {?string}
* @protected
* @instance
* @new
*/
prototype._getValue = function () {
return this.element.value;
};
/**
* Set value to the checkbox
* @method _setValue
* @param {string} value
* @member ns.widget.Checkbox
* @chainable
* @instance
* @protected
* @new
*/
prototype._setValue = function (value) {
this.element.value = value;
};
// definition
ns.widget.core.Checkbox = Checkbox;
engine.defineWidget(
"Checkbox",
"input[type='checkbox']:not(.ui-slider-switch-input):not([data-role='toggleswitch']):not(.ui-toggleswitch), " +
"input.ui-checkbox",
[],
Checkbox,
""
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
Radio = function () {
var self = this;
self._inputtype = null;
},
classes = {
radio: "ui-radio"
},
prototype = new BaseWidget();
Radio.prototype = prototype;
/**
* Build Radio widget
* @method _build
* @param {HTMLElement} element
* @protected
* @member ns.widget.Radio
* @instance
*/
prototype._build = function (element) {
var inputType = element.getAttribute("type");
if (inputType !== "radio") {
//_build should always return element
return element;
}
element.classList.add(classes.radio);
return element;
};
/**
* Returns the value of radio
* @method _getValue
* @member ns.widget.Radio
* @return {?string}
* @protected
* @instance
* @new
*/
prototype._getValue = function () {
return this.element.value;
};
/**
* Set value to the radio
* @method _setValue
* @param {string} value
* @member ns.widget.Radio
* @chainable
* @instance
* @protected
* @new
*/
prototype._setValue = function (value) {
this.element.value = value;
};
// definition
ns.widget.core.Radio = Radio;
engine.defineWidget(
"Radio",
"input[type='radio'], input.ui-radio",
[],
Radio,
""
);
}(window.document, ns));
/*global window, define, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # Marquee Text
* It makes <div> element with text move horizontally like legacy <marquee> tag
*
* ## Make Marquee Element
* If you want to use Marquee widget, you have to declare below attributes in <div> element and make Marquee widget in JS code.
* To use a Marquee widget in your application, use the following code:
*
* @example
* <div class="ui-content">
* <ul class="ui-listview">
* <li><div class="ui-marquee" id="marquee">Marquee widget code sample</div></li>
* </ul>
* </div>
* <script>
* var marqueeEl = document.getElementById("marquee"),
* marqueeWidget = new tau.widget.Marquee(marqueeEl, {marqueeStyle: "scroll", delay: "3000"});
* </script>
*
* @author Heeju Joo <heeju.joo@samsung.com>
* @class ns.widget.core.Marquee
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
/**
* Alias for class ns.engine
* @property {ns.engine} engine
* @member ns.widget.core.Marquee
* @private
*/
engine = ns.engine,
/**
* Alias for class ns.event
* @property {ns.event} event
* @member ns.widget.core.Marquee
* @private
*/
utilEvent = ns.event,
/**
* Alias for class ns.util.object
* @property {Object} objectUtils
* @member ns.widget.core.Marquee
* @private
*/
objectUtils = ns.util.object,
/**
* Alias for class ns.util.DOM
* @property {Object} domUtil
* @member ns.widget.core.Marquee
* @private
*/
domUtil = ns.util.DOM,
Marquee = function() {
this._ui = {};
this._ui.marqueeInnerElement = null;
this._ui.styleSheelElement = null;
this._state = states.STOPPED;
this._hasEllipsisText = false;
this.options = objectUtils.merge({}, Marquee.defaults);
// event callbacks
this._callbacks = {};
},
prototype = new BaseWidget(),
CLASSES_PREFIX = "ui-marquee",
states = {
RUNNING: "running",
STOPPED: "stopped",
IDLE: "idle"
},
eventType = {
/**
* Triggered when the marquee animation end.
* @event marqueeend
* @memeber ns.widget.core.Marquee
*/
MARQUEE_START: "marqueestart",
MARQUEE_END: "marqueeend",
MARQUEE_STOPPED: "marqueestopped"
},
/**
* Dictionary for CSS class of marquee play state
* @property {Object} classes
* @member ns.widget.core.Marquee
* @static
*/
classes = {
MARQUEE_CONTENT: CLASSES_PREFIX + "-content",
MARQUEE_GRADIENT: CLASSES_PREFIX + "-gradient",
MARQUEE_ELLIPSIS: CLASSES_PREFIX + "-ellipsis",
ANIMATION_RUNNING: CLASSES_PREFIX + "-anim-running",
ANIMATION_STOPPED: CLASSES_PREFIX + "-anim-stopped",
ANIMATION_IDLE: CLASSES_PREFIX + "-anim-idle"
},
selector = {
MARQUEE_CONTENT: "." + CLASSES_PREFIX + "-content"
},
/**
* Dictionary for marquee style
*/
style = {
SCROLL: "scroll",
SLIDE: "slide",
ALTERNATE: "alternate",
ENDTOEND: "endToEnd"
},
ellipsisEffect = {
GRADIENT: "gradient",
ELLIPSIS: "ellipsis",
NONE: "none"
},
/**
* Options for widget
* @property {Object} options
* @property {string|"slide"|"scroll"|"alternate"} [options.marqueeStyle="slide"] Sets the default style for the marquee
* @property {number} [options.speed=60] Sets the speed(px/sec) for the marquee
* @property {number|"infinite"} [options.iteration=1] Sets the iteration count number for marquee
* @property {number} [options.delay=2000] Sets the delay(ms) for marquee
* @property {"linear"|"ease"|"ease-in"|"ease-out"|"cubic-bezier(n,n,n,n)"} [options.timingFunction="linear"] Sets the timing function for marquee
* @property {"gradient"|"ellipsis"|"none"} [options.ellipsisEffect="gradient"] Sets the end-effect(gradient) of marquee
* @property {boolean} [options.autoRun=true] Sets the status of autoRun
* @member ns.widget.core.Marquee
* @static
*/
defaults = {
marqueeStyle: style.SLIDE,
speed: 60,
iteration: 1,
delay: 0,
timingFunction: "linear",
ellipsisEffect: "gradient",
runOnlyOnEllipsisText: true,
autoRun: true
};
Marquee.classes = classes;
Marquee.defaults = defaults;
/* Marquee AnimationEnd callback */
function marqueeEndHandler(self) {
self.reset();
}
function getAnimationDuration(self, speed) {
var marqueeInnerElement = self._ui.marqueeInnerElement,
textWidth = marqueeInnerElement.scrollWidth,
duration = textWidth / speed;
return duration;
}
function setMarqueeKeyFrame(self, marqueeStyle) {
var marqueeInnerElement = self._ui.marqueeInnerElement,
marqueeContainer = self.element,
containerWidth = marqueeContainer.offsetWidth,
textWidth = marqueeInnerElement.scrollWidth,
styleElement = document.createElement("style"),
keyFrameName = marqueeStyle + "-" + self.id,
customKeyFrame,
returnTimeFrame;
switch (marqueeStyle) {
case style.SLIDE:
customKeyFrame = "@-webkit-keyframes " + keyFrameName + " {"
+ "0% { -webkit-transform: translate3d(0, 0, 0);}"
+ "95%, 100% { -webkit-transform: translate3d(-" + (textWidth - containerWidth) + "px, 0, 0);} }";
break;
case style.SCROLL:
customKeyFrame = "@-webkit-keyframes " + keyFrameName + " {"
+ "0% { -webkit-transform: translate3d(0, 0, 0);}"
+ "95%, 100% { -webkit-transform: translate3d(-100%, 0, 0);} }";
break;
case style.ALTERNATE:
customKeyFrame = "@-webkit-keyframes " + keyFrameName + " {"
+ "0% { -webkit-transform: translate3d(0, 0, 0);}"
+ "50% { -webkit-transform: translate3d(-" + (textWidth - containerWidth) + "px, 0, 0);}"
+ "100% { -webkit-transform: translate3d(0, 0, 0);} }";
break;
case style.ENDTOEND:
returnTimeFrame = parseInt((textWidth / (textWidth + containerWidth)) * 100, 10);
customKeyFrame = "@-webkit-keyframes " + keyFrameName + " {"
+ "0% { -webkit-transform: translate3d(0, 0, 0);}"
+ returnTimeFrame + "% { -webkit-transform: translate3d(-100%, 0, 0); opacity: 1;}"
+ (returnTimeFrame+1) + "% { -webkit-transform: translate3d(-100%, 0, 0); opacity: 0; }"
+ (returnTimeFrame+2) + "% { -webkit-transform: translate3d(" + containerWidth + "px, 0, 0); opacity: 0; }"
+ (returnTimeFrame+3) + "% { -webkit-transform: translate3d(" + containerWidth + "px, 0, 0); opacity: 1; }"
+ "100% { -webkit-transform: translate3d(0, 0, 0);} }";
break;
default:
customKeyFrame = null;
break;
}
if (customKeyFrame) {
self.element.appendChild(styleElement);
styleElement.sheet.insertRule(customKeyFrame, 0);
self._ui.styleSheelElement = styleElement;
}
return keyFrameName;
}
function setAnimationStyle(self, options) {
var marqueeInnerElement = self._ui.marqueeInnerElement,
marqueeInnerElementStyle = marqueeInnerElement.style,
duration = getAnimationDuration(self, isNaN(parseInt(options.speed, 10))? defaults.speed : options.speed ),
marqueeKeyFrame = setMarqueeKeyFrame(self, options.marqueeStyle),
iteration;
// warning when option value is not correct.
if (isNaN(parseInt(options.speed, 10))) {
ns.warn("speed value must be number(px/sec)");
}
if ((options.iteration !== "infinite") && isNaN(options.iteration)) {
ns.warn("iteration count must be number or 'infinite'");
}
if (isNaN(options.delay)) {
ns.warn("delay value must be number");
}
marqueeInnerElementStyle.webkitAnimationName = marqueeKeyFrame;
marqueeInnerElementStyle.webkitAnimationDuration = duration + "s";
marqueeInnerElementStyle.webkitAnimationIterationCount = options.iteration;
marqueeInnerElementStyle.webkitAnimationTimingFunction = options.timingFunction;
marqueeInnerElementStyle.webkitAnimationDelay = options.delay + "ms";
}
function setEllipsisEffectStyle(self, ellipsisEffectOption, hasEllipsisText) {
var marqueeElement = self.element;
switch (ellipsisEffectOption) {
case ellipsisEffect.GRADIENT:
if (hasEllipsisText) {
marqueeElement.classList.add(classes.MARQUEE_GRADIENT);
}
break;
case ellipsisEffect.ELLIPSIS:
marqueeElement.classList.add(classes.MARQUEE_ELLIPSIS);
break;
default :
break;
}
}
function setAutoRunState(self, autoRunOption) {
if (autoRunOption) {
self.start();
} else {
self.stop();
}
}
/**
* Build Marquee DOM
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.core.Marquee
*/
prototype._build = function(element) {
var marqueeInnerElement = document.createElement("div");
while (element.hasChildNodes()) {
marqueeInnerElement.appendChild(element.removeChild(element.firstChild));
}
marqueeInnerElement.classList.add(classes.MARQUEE_CONTENT);
element.appendChild(marqueeInnerElement);
this._ui.marqueeInnerElement = marqueeInnerElement;
return element;
};
/**
* Init Marquee Style
* @method _init
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.core.Marquee
*/
prototype._init = function(element) {
var self = this;
self._ui.marqueeInnerElement = self._ui.marqueeInnerElement || element.querySelector(selector.MARQUEE_CONTENT);
self._hasEllipsisText = element.offsetWidth - domUtil.getCSSProperty(element, "padding-right", null, "float") < self._ui.marqueeInnerElement.scrollWidth;
if (!(self.options.runOnlyOnEllipsisText && !self._hasEllipsisText)) {
setEllipsisEffectStyle(self, self.options.ellipsisEffect, self._hasEllipsisText);
setAnimationStyle(self, self.options);
setAutoRunState(self, self.options.autoRun);
}
return element;
};
/**
* Bind events
* @method _bindEvents
* @protected
* @member ns.widget.core.Marquee
*/
prototype._bindEvents = function() {
var self = this,
marqueeInnerElement = self._ui.marqueeInnerElement,
animationEndCallback = marqueeEndHandler.bind(null, self);
self._callbacks.animationEnd = animationEndCallback;
utilEvent.one(marqueeInnerElement, "webkitAnimationEnd", animationEndCallback);
};
/**
* Refresh styles
* @method _refresh
* @protected
* @memeber ns.widget.core.Marquee
*/
prototype._refresh = function() {
var self = this;
self._resetStyle();
self._hasEllipsisText = self.element.offsetWidth < self._ui.marqueeInnerElement.scrollWidth;
if (self.options.runOnlyOnEllipsisText && !self._hasEllipsisText) {
return;
}
setEllipsisEffectStyle(self, self.options.ellipsisEffect, self._hasEllipsisText);
setAnimationStyle(self, self.options);
setAutoRunState(self, self.options.autoRun);
};
/**
* Reset style of Marquee elements
* @method _resetStyle
* @protected
* @memeber ns.widget.core.Marquee
*/
prototype._resetStyle = function() {
var self = this,
marqueeContainer = self.element,
marqueeKeyframeStyleSheet = self._ui.styleSheelElement,
marqueeInnerElementStyle = self._ui.marqueeInnerElement.style;
if (marqueeContainer.contains(marqueeKeyframeStyleSheet)) {
marqueeContainer.removeChild(marqueeKeyframeStyleSheet);
}
marqueeInnerElementStyle.webkitAnimationName = "";
marqueeInnerElementStyle.webkitAnimationDuration = "";
marqueeInnerElementStyle.webkitAnimationDelay = "";
marqueeInnerElementStyle.webkitAnimationIterationCount = "";
marqueeInnerElementStyle.webkitAnimationTimingFunction = "";
};
/**
* Remove marquee object and Reset DOM structure
* @method _resetDOM
* @protected
* @memeber ns.widget.core.Marquee
*/
prototype._resetDOM = function() {
var ui = this._ui;
while (ui.marqueeInnerElement.hasChildNodes()) {
this.element.appendChild(ui.marqueeInnerElement.removeChild(ui.marqueeInnerElement.firstChild));
}
this.element.removeChild(ui.marqueeInnerElement);
return null;
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.core.Marquee
*/
prototype._destroy = function() {
var self = this;
self._resetStyle();
self._resetDOM();
self._callbacks = null;
self._ui = null;
return null;
};
/**
* Set Marquee animation status Running
* @method _animationStart
* @memeber ns.widget.core.Marquee
*/
prototype._animationStart = function() {
var self = this,
marqueeElementClassList = self.element.classList,
marqueeInnerElementClassList = self._ui.marqueeInnerElement.classList;
self._state = states.RUNNING;
if (marqueeElementClassList.contains(classes.MARQUEE_ELLIPSIS)) {
marqueeElementClassList.remove(classes.MARQUEE_ELLIPSIS);
}
marqueeInnerElementClassList.remove(classes.ANIMATION_IDLE, classes.ANIMATION_STOPPED);
marqueeInnerElementClassList.add(classes.ANIMATION_RUNNING);
self.trigger(eventType.MARQUEE_START);
};
/**
* Start Marquee animation
*
* #####Running example in pure JavaScript:
*
* @example
* <div class="ui-marquee" id="marquee">
* <p>MarqueeTEST TEST message TEST for marquee</p>
* </div>
* <script>
* var marqueeWidget = tau.widget.Marquee(document.getElementById("marquee"));
* marqueeWidget.start();
* </script>
*
* @method start
* @memeber ns.widget.core.Marquee
*/
prototype.start = function() {
var self = this;
if (self.options.runOnlyOnEllipsisText && !self._hasEllipsisText) {
return;
}
switch (self._state) {
case states.IDLE:
setAnimationStyle(self, self.options);
self._bindEvents();
self._animationStart();
break;
case states.STOPPED:
self._state = states.RUNNING;
self._animationStart();
break;
case states.RUNNING:
break;
}
};
/**
* Pause Marquee animation
*
* #####Running example in pure JavaScript:
* @example
* <div class="ui-marquee" id="marquee">
* <p>MarqueeTEST TEST message TEST for marquee</p>
* </div>
* <script>
* var marqueeWidget = tau.widget.Marquee(document.getElementById("marquee"));
* marqueeWidget.stop();
* </script>
*
* @method stop
* @member ns.widget.core.Marquee
*/
prototype.stop = function() {
var self = this,
marqueeInnerElementClassList = self._ui.marqueeInnerElement.classList;
if (self.options.runOnlyOnEllipsisText && !self._hasEllipsisText) {
return;
}
if (self._state == states.IDLE) {
return;
}
self._state = states.STOPPED;
marqueeInnerElementClassList.remove(classes.ANIMATION_RUNNING);
marqueeInnerElementClassList.add(classes.ANIMATION_STOPPED);
self.trigger(eventType.MARQUEE_STOPPED);
};
/**
* Reset Marquee animation
*
* #####Running example in pure JavaScript:
* @example
* <div class="ui-marquee" id="marquee">
* <p>MarqueeTEST TEST message TEST for marquee</p>
* </div>
* <script>
* var marqueeWidget = tau.widget.Marquee(document.getElementById("marquee"));
* marqueeWidget.reset();
* </script>
*
* @method reset
* @member ns.widget.core.Marquee
*/
prototype.reset = function() {
var self = this,
marqueeElementClassList = self.element.classList,
marqueeInnerElementClassList = self._ui.marqueeInnerElement.classList;
if (self.options.runOnlyOnEllipsisText && !self._hasEllipsisText) {
return;
}
if (self._state == states.IDLE) {
return;
}
self._state = states.IDLE;
marqueeInnerElementClassList.remove(classes.ANIMATION_RUNNING, classes.ANIMATION_STOPPED);
marqueeInnerElementClassList.add(classes.ANIMATION_IDLE);
if (self.options.ellipsisEffect === ellipsisEffect.ELLIPSIS) {
marqueeElementClassList.add(classes.MARQUEE_ELLIPSIS);
}
self._resetStyle();
self.trigger(eventType.MARQUEE_END);
};
Marquee.prototype = prototype;
ns.widget.core.Marquee = Marquee;
engine.defineWidget(
"Marquee",
".ui-marquee",
["start", "stop", "reset"],
Marquee,
"core"
);
}(window.document, ns));
/*global window, define, XMLHttpRequest */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Load Utility
* Object contains function to load external resources.
* @class ns.util.load
*/
(function (document, ns) {
'use strict';
/**
* Local alias for document HEAD element
* @property {HTMLHeadElement} head
* @static
* @private
* @member ns.util.load
*/
var head = document.head,
/**
* Local alias for document styleSheets element
* @property {HTMLStyleElement} styleSheets
* @static
* @private
* @member ns.util.load
*/
styleSheets = document.styleSheets,
/**
* Local alias for ns.util.DOM
* @property {Object} utilsDOM Alias for {@link ns.util.DOM}
* @member ns.util.load
* @static
* @private
*/
utilDOM = ns.util.DOM,
getNSData = utilDOM.getNSData,
setNSData = utilDOM.setNSData,
load = ns.util.load || {},
/**
* Regular expression for extracting path to the image
* @property {RegExp} IMAGE_PATH_REGEXP
* @static
* @private
* @member ns.util.load
*/
IMAGE_PATH_REGEXP = /url\((\.\/)?images/gm,
/**
* Regular expression for extracting path to the css
* @property {RegExp} CSS_FILE_REGEXP
* @static
* @private
* @member ns.util.load
*/
CSS_FILE_REGEXP = /[^/]+\.css$/;
/**
* Load file
* (synchronous loading)
* @method loadFileSync
* @param {string} scriptPath
* @param {?Function} successCB
* @param {?Function} errorCB
* @static
* @private
* @member ns.util.load
*/
function loadFileSync(scriptPath, successCB, errorCB) {
var xhrObj = new XMLHttpRequest();
// open and send a synchronous request
xhrObj.open('GET', scriptPath, false);
xhrObj.send();
// add the returned content to a newly created script tag
if (xhrObj.status === 200 || xhrObj.status === 0) {
if (typeof successCB === 'function') {
successCB(xhrObj, xhrObj.status);
}
} else {
if (typeof errorCB === 'function') {
errorCB(xhrObj, xhrObj.status, new Error(xhrObj.statusText));
}
}
}
/**
* Callback function on javascript load success
* @method scriptSyncSuccess
* @private
* @static
* @param {?Function} successCB
* @param {?Function} xhrObj
* @param {?string} status
* @member ns.util.load
*/
function scriptSyncSuccess(successCB, xhrObj, status) {
var script = document.createElement('script');
script.type = 'text/javascript';
script.text = xhrObj.responseText;
document.body.appendChild(script);
if (typeof successCB === 'function') {
successCB(xhrObj, status);
}
}
/**
* Add script to document
* (synchronous loading)
* @method scriptSync
* @param {string} scriptPath
* @param {?Function} successCB
* @param {?Function} errorCB
* @static
* @member ns.util.load
*/
function scriptSync(scriptPath, successCB, errorCB) {
loadFileSync(scriptPath, scriptSyncSuccess.bind(null, successCB), errorCB);
}
/**
* Callback function on css load success
* @method cssSyncSuccess
* @param {string} cssPath
* @param {?Function} successCB
* @param {?Function} xhrObj
* @member ns.util.load
* @static
* @private
*/
function cssSyncSuccess(cssPath, successCB, xhrObj) {
var css = document.createElement('style');
css.type = 'text/css';
css.textContent = xhrObj.responseText.replace(
IMAGE_PATH_REGEXP,
'url(' + cssPath.replace(CSS_FILE_REGEXP, 'images')
);
if (typeof successCB === 'function') {
successCB(css);
}
}
/**
* Add css to document
* (synchronous loading)
* @method cssSync
* @param {string} cssPath
* @param {?Function} successCB
* @param {?Function} errorCB
* @static
* @private
* @member ns.util.load
*/
function cssSync(cssPath, successCB, errorCB) {
loadFileSync(cssPath, cssSyncSuccess.bind(null, cssPath, successCB), errorCB);
}
/**
* Add element to head tag
* @method addElementToHead
* @param {HTMLElement} element
* @param {boolean} [asFirstChildElement=false]
* @member ns.util.load
* @static
*/
function addElementToHead(element, asFirstChildElement) {
var firstElement;
if (head) {
if (asFirstChildElement) {
firstElement = head.firstElementChild;
if (firstElement) {
head.insertBefore(element, firstElement);
return;
}
}
head.appendChild(element);
}
}
/**
* Create HTML link element with href
* @method makeLink
* @param {string} href
* @returns {HTMLLinkElement}
* @member ns.util.load
* @static
*/
function makeLink(href) {
var cssLink = document.createElement('link');
cssLink.setAttribute('rel', 'stylesheet');
cssLink.setAttribute('href', href);
cssLink.setAttribute('name', 'tizen-theme');
return cssLink;
}
/**
* Adds the given node to document head or replaces given 'replaceElement'.
* Additionally adds 'name' and 'theme-name' attribute
* @param {HTMLElement} node Element to be placed as theme link
* @param {string} themeName Theme name passed to the element
* @param {HTMLElement} [replaceElement=null] If replaceElement is given it gets replaced by node
*/
function addNodeAsTheme(node, themeName, replaceElement) {
setNSData(node, 'name', 'tizen-theme');
setNSData(node, 'theme-name', themeName);
if (replaceElement) {
replaceElement.parentNode.replaceChild(node, replaceElement);
} else {
addElementToHead(node, true);
}
}
/**
* Add css link element to head if not exists
* @method themeCSS
* @param {string} path
* @param {string} themeName
* @param {boolean} [embed=false] Embeds the CSS content to the document
* @member ns.util.load
* @static
*/
function themeCSS(path, themeName, embed) {
var i,
styleSheetsLength = styleSheets.length,
ownerNode,
previousElement = null,
linkElement;
// Find css link or style elements
for (i = 0; i < styleSheetsLength; i++) {
ownerNode = styleSheets[i].ownerNode;
// We try to find a style / link node that matches current style or is linked to
// the proper theme. We cannot use ownerNode.href because this returns the absolute path
if (getNSData(ownerNode, 'name') === 'tizen-theme' || ownerNode.getAttribute("href") === path) {
if (getNSData(ownerNode, 'theme-name') === themeName) {
// Nothing to change
return;
}
previousElement = ownerNode;
break;
}
}
if (embed){
// Load and replace old styles or append new styles
cssSync(path, function onSuccess(styleElement) {
addNodeAsTheme(styleElement, themeName, previousElement);
}, function onFailure(xhrObj, xhrStatus, errorObj) {
ns.warn("There was a problem when loading '" + themeName + "', status: " + xhrStatus);
});
} else {
linkElement = makeLink(path);
addNodeAsTheme(linkElement, themeName, previousElement);
}
}
/**
* In debug mode add time to url to disable cache
* @property {string} cacheBust
* @member ns.util.load
* @static
*/
load.cacheBust = (document.location.href.match(/debug=true/)) ? '?cacheBust=' + (new Date()).getTime() : '';
// the binding a local methods with the namespace
load.scriptSync = scriptSync;
load.addElementToHead = addElementToHead;
load.makeLink = makeLink;
load.themeCSS = themeCSS;
ns.util.load = load;
}(window.document, ns));
/*global window, define, Math, ns*/
/*jslint bitwise: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Theme object
* Class with functions to set theme of application.
* @class ns.theme
*/
(function (window, document, ns) {
"use strict";
/**
* Local alias for document HEAD element
* @property {HTMLHeadElement} head
* @static
* @private
* @member ns.theme
*/
var head = document.head,
documentElement = document.documentElement,
frameworkData = ns.frameworkData,
util = ns.util,
DOM = util.DOM,
load = util.load,
support = ns.support,
stopEvent = function (event) {
var element = event.target,
tag = element.tagName.toLowerCase(),
type = element.type;
if ((tag !== "input" ||
(type !== "text" && type !== "email" && type !== "url" && type !== "search" && type !== "tel")) &&
tag !== "textarea") {
event.stopPropagation();
event.preventDefault();
}
},
THEME_JS_FILE_NAME = "theme.js",
THEME_CSS_FILE_NAME = "tau",
themeRegex = /ui-(bar|body|overlay)-([a-z])\b/,
deviceWidthRegex = /.*width=(device-width|\d+)\s*,?.*$/gi;
ns.theme = {
/**
* Standard theme
* @property {string} theme="s"
* @member ns.theme
*/
theme: "s",
_activeTheme: null,
/**
* This function inits theme.
* @method init
* @param {HTMLElement} container
* @member ns.theme
*/
init: function (container) {
var self = this,
containerClassList = container.classList;
if (frameworkData) {
frameworkData.getParams();
}
if (support && support.gradeA()) {
documentElement.classList.add("ui-mobile");
containerClassList.add("ui-mobile-viewport");
}
if (frameworkData) {
self.loadTheme(frameworkData.theme);
}
},
/**
* This function scales font size.
* @method scaleBaseFontSize
* @param {number} themeDefaultFontSize Default font size
* @param {number} ratio Scaling ration
* @member ns.theme
*/
scaleBaseFontSize : function (themeDefaultFontSize, ratio) {
var scaledFontSize = Math.max(themeDefaultFontSize * ratio | 0, 4);
documentElement.style.fontSize = scaledFontSize + "px";
document.body.style.fontSize = scaledFontSize + "px";
},
/**
* This function searches theme, which is inherited
* from parents by element.
* @method getInheritedTheme
* @param {HTMLElement} element Element for which theme is looking for.
* @param {string} defaultTheme Default theme.
* It is used if no theme, which can be inherited, is found.
* @return {string} Inherited theme
* @member ns.theme
*/
getInheritedTheme: function (element, defaultTheme) {
var theme,
parentElement = element.parentNode,
parentClasses,
parentTheme;
theme = DOM.getNSData(element, "theme");
if (!theme) {
while (parentElement) {
parentClasses = parentElement.className || "";
parentTheme = themeRegex.exec(parentClasses);
if (parentClasses && parentTheme && parentTheme.length > 2) {
theme = parentTheme[2];
break;
}
parentElement = parentElement.parentNode;
}
}
return theme || defaultTheme;
},
/**
* This function sets selection behavior for the element.
* @method enableSelection
* @param {HTMLElement} element Element for which selection behavior is set.
* @param {"text"|"auto"|"none"} value="auto" Selection behavior.
* @return {HTMLElement} Element with set styles.
* @member ns.theme
*/
enableSelection: function (element, value) {
var val,
elementStyle;
switch (value) {
case "text":
case "auto":
case "none":
val = value;
break;
default:
val = "auto";
break;
}
if (element === document) {
element = document.body;
}
elementStyle = element.style;
elementStyle.MozUserSelect = elementStyle.webkitUserSelect = elementStyle.userSelect = val;
return element;
},
/**
* This function disables event "contextmenu".
* @method disableContextMenu
* @param {HTMLElement} element Element for which event "contextmenu"
* is disabled.
* @member ns.theme
*/
disableContextMenu: function (element) {
element.addEventListener("contextmenu", stopEvent, true);
},
/**
* This function enables event "contextmenu".
* @method enableContextMenu
* @param {HTMLElement} element Element for which event "contextmenu"
* is enabled.
* @member ns.theme
*/
enableContextMenu: function (element) {
element.removeEventListener("contextmenu", stopEvent, true);
},
/**
* This function loads files with proper theme.
* @method loadTheme
* @param {string} theme Choosen theme.
* @member ns.theme
*/
loadTheme: function(theme) {
var self = this,
themePath = frameworkData.themePath,
themeName = THEME_CSS_FILE_NAME,
cssPath,
isMinified = frameworkData.minified,
jsPath;
// If the theme has been loaded do not repeat that process
if (frameworkData.themeLoaded) {
return;
}
if (frameworkData.frameworkName !== "tau") {
themeName = "tizen-web-ui-fw-theme";
}
if (isMinified) {
cssPath = themePath + "/" + themeName + ".min.css";
} else {
cssPath = themePath + "/" + themeName + ".css";
}
load.themeCSS(cssPath, theme);
jsPath = themePath + "/" + THEME_JS_FILE_NAME;
load.scriptSync(jsPath);
if (support.gradeA()) {
self.setScaling();
}
frameworkData.themeLoaded = true;
},
/**
* This function sets viewport.
* If custom viewport is found, its width will be returned.
* Otherwise, the new viewport will be created.
* @method setViewport
* @param {number|string} viewportWidth Width of the new viewport.
* If no viewport is found, the new viewport with this
* width is created.
* @return {string} Width of custom viewport.
* @member ns.theme
*/
setViewport: function(viewportWidth) {
var metaViewport = document.querySelector("meta[name=viewport]"),
content;
if (metaViewport) {
// Found custom viewport!
content = metaViewport.getAttribute("content");
viewportWidth = content.replace(deviceWidthRegex, "$1");
} else {
// Create a meta tag
metaViewport = document.createElement("meta");
metaViewport.name = "viewport";
content = "width=" + viewportWidth + ", user-scalable=no";
metaViewport.content = content;
head.insertBefore(metaViewport, head.firstChild);
}
return viewportWidth;
},
/**
* This function checks if application is run
* in the mobile browser.
* @method isMobileBrowser
* @return {boolean} Returns true, if application
* is run in mobile browser. Otherwise, false is returned.
* @member ns.theme
*/
isMobileBrowser: function() {
return window.navigator.appVersion.indexOf("Mobile") > -1;
},
/**
* This function sets scaling of viewport.
* @method setScaling
* @member ns.theme
*/
setScaling: function () {
var self = this,
viewportWidth = frameworkData.viewportWidth,
themeDefaultFontSize = frameworkData.defaultFontSize, // comes from theme.js
ratio = 1;
// Keep original font size
document.body.setAttribute("data-tizen-theme-default-font-size", themeDefaultFontSize);
if (ns.theme.isMobileBrowser()) {
// Legacy support: tizen.frameworkData.viewportScale
if (frameworkData.viewportScale === true) {
viewportWidth = "screen-width";
}
// screen-width support
if ("screen-width" === viewportWidth) {
if (window.self === window.top) {
// Top frame: for target. Use window.outerWidth.
viewportWidth = window.outerWidth;
} else {
// iframe: for web simulator. Use clientWidth.
viewportWidth = document.documentElement.clientWidth;
}
}
// set viewport meta tag
// If custom viewport setting exists, get viewport width
viewportWidth = self.setViewport(viewportWidth);
if (viewportWidth !== "device-width") {
ratio = parseFloat(viewportWidth / ns.frameworkData.defaultViewportWidth);
self.scaleBaseFontSize(themeDefaultFontSize, ratio);
}
}
}
};
document.addEventListener("themeinit", function (evt) {
var router = evt.detail;
if (router && ns.getConfig("autoInitializePage", true)) {
ns.theme.init(router.getContainer().element);
}
}, false);
}(window, window.document, ns));
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Gesture Namespace
* Core object enables multi gesture support.
*
* @class ns.event.gesture
*/
(function ( ns, window, undefined ) {
"use strict";
var Gesture = function( elem, options ) {
return new ns.event.gesture.Instance( elem, options );
};
/**
* Default values for Gesture feature
* @property {Object} defaults
* @property {boolean} [defaults.triggerEvent=false]
* @property {number} [defaults.updateVelocityInterval=16]
* Interval in which Gesture recalculates current velocity in ms
* @property {number} [defaults.estimatedPointerTimeDifference=15]
* pause time threshold.. tune the number to up if it is slow
* @member ns.event.gesture
* @static
*/
Gesture.defaults = {
triggerEvent: false,
updateVelocityInterval: 16,
estimatedPointerTimeDifference: 15
};
/**
* Dictionary of orientation
* @property {Object} Orientation
* @property {1} Orientation.VERTICAL vertical orientation
* @property {2} Orientation.HORIZONTAL horizontal orientation
* @member ns.event.gesture
* @static
*/
Gesture.Orientation = {
VERTICAL: "vertical",
HORIZONTAL: "horizontal"
};
/**
* Dictionary of direction
* @property {Object} Direction
* @property {1} Direction.UP up
* @property {2} Direction.DOWN down
* @property {3} Direction.LEFT left
* @property {4} Direction.RIGHT right
* @member ns.event.gesture
* @static
*/
Gesture.Direction = {
UP: "up",
DOWN: "down",
LEFT: "left",
RIGHT: "right"
};
/**
* Dictionary of gesture events state
* @property {Object} Event
* @property {"start"} Event.START start
* @property {"move"} Event.MOVE move
* @property {"end"} Event.END end
* @property {"cancel"} Event.CANCEL cancel
* @property {"blocked"} Event.BLOCKED blocked
* @member ns.event.gesture
* @static
*/
Gesture.Event = {
START: "start",
MOVE: "move",
END: "end",
CANCEL: "cancel",
BLOCKED: "blocked"
};
/**
* Dictionary of gesture events flags
* @property {Object} Result
* @property {number} [Result.PENDING=1] is pending
* @property {number} [Result.RUNNING=2] is running
* @property {number} [Result.FINISHED=4] is finished
* @property {number} [Result.BLOCK=8] is blocked
* @member ns.event.gesture
* @static
*/
Gesture.Result = {
PENDING: 1,
RUNNING: 2,
FINISHED: 4,
BLOCK: 8
};
/**
* Create plugin namespace.
* @property {Object} plugin
* @member ns.event.gesture
* @static
*/
Gesture.plugin = {};
/**
* Create object of Detector
* @method createDetector
* @param {string} gesture
* @param {HTMLElement} eventSender
* @param {Object} options
* @return {ns.event.gesture.Gesture}
* @member ns.event.gesture
* @static
*/
Gesture.createDetector = function( gesture, eventSender, options ) {
if ( !Gesture.plugin[gesture] ) {
throw gesture + " gesture is not supported";
}
return new Gesture.plugin[gesture]( eventSender, options );
};
ns.event.gesture = Gesture;
} ( ns, window ) );
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Gesture Utilities
* Contains helper function to gesture support.
* @class ns.event.gesture.utils
*/
(function (ns, Math, undefined) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.utils
* @private
* @static
*/
var Gesture = ns.event.gesture;
Gesture.utils = {
/**
* Get center from array of touches
* @method getCenter
* @param {Event[]} touches description
* @member ns.event.gesture.utils
* @return {Object} position
* @return {number} return.clientX position X
* @return {number} return.clientY position Y
*/
getCenter: function (touches) {
var valuesX = [], valuesY = [];
[].forEach.call(touches, function(touch) {
// I prefer clientX because it ignore the scrolling position
valuesX.push(!isNaN(touch.clientX) ? touch.clientX : touch.pageX);
valuesY.push(!isNaN(touch.clientY) ? touch.clientY : touch.pageY);
});
return {
clientX: (Math.min.apply(Math, valuesX) + Math.max.apply(Math, valuesX)) / 2,
clientY: (Math.min.apply(Math, valuesY) + Math.max.apply(Math, valuesY)) / 2
};
},
/**
* Get velocity
* @method getVelocity
* @param {number} delta_time Delta of time
* @param {number} delta_x Position change on x axis
* @param {number} delta_y Position change on y axis
* @return {Object} velocity
* @return {number} return.x velocity on X axis
* @return {number} return.y velocity on Y axis
* @member ns.event.gesture.utils
*/
getVelocity: function (delta_time, delta_x, delta_y) {
return {
x: Math.abs(delta_x / delta_time) || 0,
y: Math.abs(delta_y / delta_time) || 0
};
},
/**
* Get angel between position of two touches
* @method getAngle
* @param {Event} touch1 first touch
* @param {Event} touch2 second touch
* @return {number} angel (deg)
* @member ns.event.gesture.utils
*/
getAngle: function (touch1, touch2) {
var y = touch2.clientY - touch1.clientY,
x = touch2.clientX - touch1.clientX;
return Math.atan2(y, x) * 180 / Math.PI;
},
/**
* Get direction indicated by position of two touches
* @method getDirectiqon
* @param {Event} touch1 first touch
* @param {Event} touch2 second touch
* @return {ns.event.gesture.Direction.LEFT|ns.event.gesture.Direction.RIGHT|ns.event.gesture.Direction.UP|ns.event.gesture.Direction.DOWN}
* @member ns.event.gesture.utils
*/
getDirection: function (touch1, touch2) {
var x = Math.abs(touch1.clientX - touch2.clientX),
y = Math.abs(touch1.clientY - touch2.clientY);
if (x >= y) {
return touch1.clientX - touch2.clientX > 0 ? Gesture.Direction.LEFT : Gesture.Direction.RIGHT;
}
return touch1.clientY - touch2.clientY > 0 ? Gesture.Direction.UP : Gesture.Direction.DOWN;
},
/**
* Get distance indicated by position of two touches
* @method getDistance
* @param {Event} touch1 first touch
* @param {Event} touch2 second touch
* @return {number} distance
* @member ns.event.gesture.utils
*/
getDistance: function (touch1, touch2) {
var x = touch2.clientX - touch1.clientX,
y = touch2.clientY - touch1.clientY;
return Math.sqrt((x * x) + (y * y));
},
/**
* Get scale indicated by position of the first and the last touch
* @method getScale
* @param {Event} start start touch
* @param {Event} end end touch
* @return {number} scale
* @member ns.event.gesture.utils
*/
getScale: function (start, end) {
// need two fingers...
if (start.length >= 2 && end.length >= 2) {
return this.getDistance(end[0], end[1]) / this.getDistance(start[0], start[1]);
}
return 1;
},
/**
* Get value of rotation indicated by position
* of the first and the last touch
* @method getRotation
* @param {Event} start start touch
* @param {Event} end end touch
* @return {number} angle (deg)
* @member ns.event.gesture.utils
*/
getRotation: function (start, end) {
// need two fingers
if (start.length >= 2 && end.length >= 2) {
return this.getAngle(end[1], end[0]) -
this.getAngle(start[1], start[0]);
}
return 0;
},
/**
* Check if the direction is vertical
* @method isVertical
* @param {ns.event.gesture.Direction.LEFT|ns.event.gesture.Direction.RIGHT|ns.event.gesture.Direction.UP|ns.event.gesture.Direction.DOWN} direction start touch
* @return {boolean}
* @member ns.event.gesture.utils
*/
isVertical: function (direction) {
return direction === Gesture.Direction.UP || direction === Gesture.Direction.DOWN;
},
/**
* Check if the direction is horizontal
* @method isHorizontal
* @param {ns.event.gesture.Direction.LEFT|ns.event.gesture.Direction.RIGHT|ns.event.gesture.Direction.UP|ns.event.gesture.Direction.DOWN} direction start touch
* @return {boolean}
* @member ns.event.gesture.utils
*/
isHorizontal: function (direction) {
return direction === Gesture.Direction.LEFT || direction === Gesture.Direction.RIGHT;
},
/**
* Check if the direction is horizontal
* @method getOrientation
* @param {ns.event.gesture.Direction.LEFT|ns.event.gesture.Direction.RIGHT|ns.event.gesture.Direction.UP|ns.event.gesture.Direction.DOWN} direction
* @return {boolean}
* @member ns.event.gesture.utils
*/
getOrientation: function (direction) {
return this.isVertical(direction) ? Gesture.Orientation.VERTICAL : Gesture.Orientation.HORIZONTAL;
}
};
} (ns, window.Math));
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Gesture.Detector class
* Base class for create detectors in gestures.
*
* @class ns.event.gesture.Detector
*/
( function ( ns, window, undefined ) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.Manager
* @private
* @static
*/
var Gesture = ns.event.gesture,
/**
* Alias for method {@link ns.util.object.merge}
* @property {Function} objectMerge
* @member ns.event.gesture.Detector
* @private
* @static
*/
objectMerge = ns.util.object.merge,
Detector = function( strategy, sender ) {
this.sender = sender;
this.strategy = strategy.create();
this.name = this.strategy.name;
this.index = this.strategy.index || 100;
this.options = this.strategy.options || {};
};
/**
* Start of gesture detection of given type
* @method detect
* @param {string} gestureEvent
* @return {Object}
* @member ns.event.gesture.Detector
*/
Detector.prototype.detect = function( gestureEvent ) {
return this.strategy.handler( gestureEvent, this.sender, this.strategy.options );
};
Detector.Sender = {
sendEvent: function(/* eventName, detail */) {}
};
/**
* Create plugin namespace.
* @property {Object} plugin
* @member ns.event.gesture.Detector
*/
Detector.plugin = {};
/**
* Methods creates plugin
* @method create
* @param {Object} gestureHandler
* @return {ns.event.gesture.Detector} gestureHandler
* @member ns.event.gesture.Detector.plugin
*/
Detector.plugin.create = function( gestureHandler ) {
if ( !gestureHandler.types ) {
gestureHandler.types = [ gestureHandler.name ];
}
var detector = Detector.plugin[ gestureHandler.name ] = function( options ) {
this.options = objectMerge({}, gestureHandler.defaults, options);
};
detector.prototype.create = function() {
return objectMerge({
options: this.options
}, gestureHandler);
};
return detector;
};
// definition
Gesture.Detector = Detector;
} ( ns, window ));
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Gesture.Manager class
* Main class controls all gestures.
* @class ns.event.gesture.Manager
*/
( function ( ns, window, document) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.Manager
* @private
* @static
*/
var Gesture = ns.event.gesture,
/**
* Alias for method {@link ns.util.object.merge}
* @property {Function} objectMerge
* @member ns.event.gesture.Manager
* @private
* @static
*/
objectMerge = ns.util.object.merge,
/**
* Device has touchable interface
* @property {boolean} TOUCH_DEVICE
* @member ns.event.gesture.Manager
* @private
* @static
*/
TOUCH_DEVICE = "ontouchstart" in window;
Gesture.Manager = (function() {
var instance = null,
startEvent = null,
isReadyDetecting = false,
blockMouseEvent = false,
Manager = function() {
this.instances = [];
this.gestureDetectors = [];
this.runningDetectors = [];
this.detectorRequestedBlock = null;
this.unregisterBlockList = [];
this.gestureEvents = null;
this.velocity = null;
};
Manager.prototype = {
/**
* Bind start events
* @method _bindStartEvents
* @param {ns.event.gesture.Instance} instance gesture instance
* @member ns.event.gesture.Manager
* @protected
*/
_bindStartEvents: function( instance ) {
var element = instance.getElement();
if ( TOUCH_DEVICE ) {
element.addEventListener( "touchstart", this);
}
element.addEventListener( "mousedown", this);
},
/**
* Bind move, end and cancel events
* @method _bindEvents
* @member ns.event.gesture.Manager
* @protected
*/
_bindEvents: function( ) {
if ( TOUCH_DEVICE ) {
document.addEventListener( "touchmove", this);
document.addEventListener( "touchend", this);
document.addEventListener( "touchcancel", this);
}
document.addEventListener( "mousemove", this);
document.addEventListener( "mouseup", this);
},
/**
* Unbind start events
* @method _unbindStartEvents
* @param {ns.event.gesture.Instance} instance gesture instance
* @member ns.event.gesture.Manager
* @protected
*/
_unbindStartEvents: function( instance ) {
var element = instance.getElement();
if ( TOUCH_DEVICE ) {
element.removeEventListener( "touchstart", this);
}
element.removeEventListener( "mousedown", this);
},
/**
* Unbind move, end and cancel events
* @method _bindEvents
* @member ns.event.gesture.Manager
* @protected
*/
_unbindEvents: function() {
if ( TOUCH_DEVICE ) {
document.removeEventListener( "touchmove", this);
document.removeEventListener( "touchend", this);
document.removeEventListener( "touchcancel", this);
}
document.removeEventListener( "mousemove", this);
document.removeEventListener( "mouseup", this);
},
/**
* Handle event
* @method handleEvent
* @param {Event} event
* @member ns.event.gesture.Manager
* @protected
*/
/* jshint -W086 */
handleEvent: function( event ) {
var eventType = event.type.toLowerCase();
if ( eventType.match(/touch/) ) {
blockMouseEvent = true;
}
if ( eventType.match(/mouse/) &&
( blockMouseEvent || event.which !== 1 ) ) {
return;
}
switch ( event.type ) {
case "mousedown":
case "touchstart":
this._start( event );
break;
case "mousemove":
case "touchmove":
this._move( event );
break;
case "mouseup":
case "touchend":
this._end( event );
break;
case "touchcancel":
this._cancel( event );
break;
}
},
/**
* Handler for gesture start
* @method _start
* @param {Event} event
* @member ns.event.gesture.Manager
* @protected
*/
_start: function( event ) {
var elem = event.currentTarget,
startEvent = {},
detectors = [];
if ( !isReadyDetecting ) {
this._resetDetecting();
this._bindEvents();
startEvent = this._createDefaultEventData( Gesture.Event.START, event );
this.gestureEvents = {
start: startEvent,
last: startEvent
};
this.velocity = {
event: startEvent,
x: 0,
y: 0
};
startEvent = objectMerge(startEvent, this._createGestureEvent(Gesture.Event.START, event));
isReadyDetecting = true;
}
this.instances.forEach(function( instance ) {
if ( instance.getElement() === elem ) {
detectors = detectors.concat( instance.getGestureDetectors() );
}
}, this);
detectors.sort(function(a, b) {
if(a.index < b.index) {
return -1;
} else if(a.index > b.index) {
return 1;
}
return 0;
});
this.gestureDetectors = this.gestureDetectors.concat( detectors );
this._detect(detectors, startEvent);
},
/**
* Handler for gesture move
* @method _move
* @param {Event} event
* @member ns.event.gesture.Manager
* @protected
*/
_move: function( event ) {
if ( !isReadyDetecting ) {
return;
}
event = this._createGestureEvent(Gesture.Event.MOVE, event);
this._detect(this.gestureDetectors, event);
this.gestureEvents.last = event;
},
/**
* Handler for gesture end
* @method _end
* @param {Event} event
* @member ns.event.gesture.Manager
* @protected
*/
_end: function( event ) {
event = objectMerge(
{},
this.gestureEvents.last,
this._createDefaultEventData(Gesture.Event.END, event)
);
if ( event.pointers.length > 0 ) {
return;
}
this._detect(this.gestureDetectors, event);
this.unregisterBlockList.forEach(function( instance ) {
this.unregist( instance );
}, this);
this._resetDetecting();
blockMouseEvent = false;
},
/**
* Handler for gesture cancel
* @method _cancel
* @param {Event} event
* @member ns.event.gesture.Manager
* @protected
*/
_cancel: function( event ) {
event = objectMerge(
{},
this.gestureEvents.last,
this._createDefaultEventData(Gesture.Event.CANCEL, event)
);
this._detect(this.gestureDetectors, event);
this.unregisterBlockList.forEach(function( instance ) {
this.unregist( instance );
}, this);
this._resetDetecting();
blockMouseEvent = false;
},
/**
* Detect gesture
* @method _detect
* @param {Event} event
* @member ns.event.gesture.Manager
* @protected
*/
_detect: function( detectors, event ) {
var finishedDetectors = [];
detectors.forEach(function( detector ) {
var result;
if ( this.detectorRequestedBlock ) {
return;
}
result = detector.detect( event );
if ( result & Gesture.Result.RUNNING ) {
if ( this.runningDetectors.indexOf( detector ) < 0 ) {
this.runningDetectors.push( detector );
}
}
if ( result & Gesture.Result.FINISHED ) {
finishedDetectors.push( detector );
}
if ( result & Gesture.Result.BLOCK ) {
this.detectorRequestedBlock = detector;
}
}, this);
// remove finished detectors.
finishedDetectors.forEach(function( detector ) {
var idx;
idx = this.gestureDetectors.indexOf( detector );
if ( idx > -1 ) {
this.gestureDetectors.splice(idx, 1);
}
idx = this.runningDetectors.indexOf( detector );
if ( idx > -1 ) {
this.runningDetectors.splice(idx, 1);
}
}, this);
// remove all detectors except the detector that return block result
if ( this.detectorRequestedBlock ) {
// send to cancel event.
this.runningDetectors.forEach(function( detector ) {
var cancelEvent = objectMerge({}, event, {
eventType: Gesture.Event.BLOCKED
});
detector.detect( cancelEvent );
});
this.runningDetectors.length = 0;
// remove all detectors.
this.gestureDetectors.length = 0;
if ( finishedDetectors.indexOf( this.detectorRequestedBlock ) < 0 ) {
this.gestureDetectors.push( this.detectorRequestedBlock );
}
}
},
/**
* Reset of gesture manager detector
* @method _resetDetecting
* @member ns.event.gesture.Manager
* @protected
*/
_resetDetecting: function() {
isReadyDetecting = false;
startEvent = null
this.gestureDetectors.length = 0;
this.runningDetectors.length = 0;
this.detectorRequestedBlock = null;
this.gestureEvents = null;
this.velocity = null;
this._unbindEvents();
},
/**
* Create default event data
* @method _createDefaultEventData
* @param {string} type event type
* @param {Event} event source event
* @return {Object} default event data
* @return {string} return.eventType
* @return {number} return.timeStamp
* @return {Touch} return.pointer
* @return {TouchList} return.pointers
* @return {Event} return.srcEvent
* @return {Function} return.preventDefault
* @return {Function} return.stopPropagation
* @member ns.event.gesture.Manager
* @protected
*/
_createDefaultEventData: function( type, event ) {
var pointers = event.touches ?
event.touches :
event.type === "mouseup" ? [] : ( event.identifier=1 && [event] ),
pointer = pointers[0],
timeStamp = new Date().getTime();
return {
eventType: type,
timeStamp: timeStamp,
pointer: pointer,
pointers: pointers,
srcEvent: event,
preventDefault: function() {
this.srcEvent.preventDefault();
},
stopPropagation: function() {
this.srcEvent.stopPropagation();
}
};
},
/**
* Create gesture event
* @method _createGestureEvent
* @param {string} type event type
* @param {Event} event source event
* @return {Object} gesture event consist from Event class and additional properties
* @return {number} return.deltaTime
* @return {number} return.deltaX
* @return {number} return.deltaY
* @return {number} return.velocityX
* @return {number} return.velocityY
* @return {number} return.estimatedX
* @return {number} return.estimatedY
* @return {number} return.estimatedDeltaX
* @return {number} return.estimatedDeltaY
* @return {number} return.distance
* @return {number} return.angle
* @return {ns.event.gesture.Direction.LEFT|ns.event.gesture.Direction.RIGHT|ns.event.gesture.Direction.UP|ns.event.gesture.Direction.DOWN} return.direction
* @return {number} return.scale
* @return {number} return.rotation (deg)
* @return {Event} return.startEvent
* @return {Event} return.lastEvent
* @member ns.event.gesture.Manager
* @protected
*/
_createGestureEvent: function( type, event ) {
var ev = this._createDefaultEventData( type, event ),
startEvent = this.gestureEvents.start,
lastEvent = this.gestureEvents.last,
velocityEvent = this.velocity.event,
delta = {
time: ev.timeStamp - startEvent.timeStamp,
x: ev.pointer.clientX - startEvent.pointer.clientX,
y: ev.pointer.clientY - startEvent.pointer.clientY
},
deltaFromLast = {
x: ev.pointer.clientX - lastEvent.pointer.clientX,
y: ev.pointer.clientY - lastEvent.pointer.clientY
},
velocity = this.velocity,
timeDifference = Gesture.defaults.estimatedPointerTimeDifference, /* pause time threshold.util. tune the number to up if it is slow */
estimated;
// reset start event for multi touch
if( startEvent && ev.pointers.length !== startEvent.pointers.length ) {
startEvent.pointers = [];
[].forEach.call(ev.pointers, function( pointer ) {
startEvent.pointers.push( objectMerge({}, pointer) );
});
}
if ( ev.timeStamp - velocityEvent.timeStamp > Gesture.defaults.updateVelocityInterval ) {
velocity = Gesture.utils.getVelocity(
ev.timeStamp - velocityEvent.timeStamp,
ev.pointer.clientX - velocityEvent.pointer.clientX,
ev.pointer.clientY - velocityEvent.pointer.clientY
);
objectMerge(this.velocity, velocity, {
event: ev
});
}
estimated = {
x: Math.round( ev.pointer.clientX + ( timeDifference * velocity.x * (deltaFromLast.x < 0 ? -1 : 1) ) ),
y: Math.round( ev.pointer.clientY + ( timeDifference * velocity.y * (deltaFromLast.y < 0 ? -1 : 1) ) )
};
// Prevent that point goes back even though direction is not changed.
if ( (deltaFromLast.x < 0 && estimated.x > lastEvent.estimatedX) ||
(deltaFromLast.x > 0 && estimated.x < lastEvent.estimatedX) ) {
estimated.x = lastEvent.estimatedX;
}
if ( (deltaFromLast.y < 0 && estimated.y > lastEvent.estimatedY) ||
(deltaFromLast.y > 0 && estimated.y < lastEvent.estimatedY) ) {
estimated.y = lastEvent.estimatedY;
}
objectMerge(ev, {
deltaTime: delta.time,
deltaX: delta.x,
deltaY: delta.y,
velocityX: velocity.x,
velocityY: velocity.y,
estimatedX: estimated.x,
estimatedY: estimated.y,
estimatedDeltaX: estimated.x - startEvent.pointer.clientX,
estimatedDeltaY: estimated.y - startEvent.pointer.clientY,
distance: Gesture.utils.getDistance(startEvent.pointer, ev.pointer),
angle: Gesture.utils.getAngle(startEvent.pointer, ev.pointer),
direction: Gesture.utils.getDirection(startEvent.pointer, ev.pointer),
scale: Gesture.utils.getScale(startEvent.pointers, ev.pointers),
rotation: Gesture.utils.getRotation(startEvent.pointers, ev.pointers),
startEvent: startEvent,
lastEvent: lastEvent
});
return ev;
},
/**
* Register instance of gesture
* @method register
* @param {ns.event.gesture.Instance} instance gesture instance
* @member ns.event.gesture.Manager
*/
register: function( instance ) {
var idx = this.instances.indexOf( instance );
if ( idx < 0 ) {
this.instances.push( instance );
this._bindStartEvents( instance );
}
},
/**
* Unregister instance of gesture
* @method unregister
* @param {ns.event.gesture.Instance} instance gesture instance
* @member ns.event.gesture.Manager
*/
unregister: function( instance ) {
var idx;
if ( !!this.gestureDetectors.length ) {
this.unregisterBlockList.push( instance );
return;
}
idx = this.instances.indexOf( instance );
if ( idx > -1 ) {
this.instances.splice( idx, 1 );
this._unbindStartEvents( instance );
}
if ( !this.instances.length ) {
this._destroy();
}
},
/**
* Destroy instance of Manager
* @method _destroy
* @member ns.event.gesture.Manager
* @protected
*/
_destroy: function() {
this._resetDetecting();
this.instances.length = 0;
this.unregisterBlockList.length = 0;
blockMouseEvent = false;
instance = null;
}
};
return {
getInstance: function() {
return instance ? instance : ( instance = new Manager() );
}
};
})();
} ( ns, window, window.document ) );
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Gesture.Instance class
* Creates instance of gesture manager on element.
* @class ns.event.gesture.Instance
*/
( function ( ns, window, undefined ) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.Instance
* @private
* @static
*/
var Gesture = ns.event.gesture,
/**
* Local alias for {@link ns.event.gesture.Detector}
* @property {Object}
* @member ns.event.gesture.Instance
* @private
* @static
*/
Detector = ns.event.gesture.Detector,
/**
* Local alias for {@link ns.event.gesture.Manager}
* @property {Object}
* @member ns.event.gesture.Instance
* @private
* @static
*/
Manager = ns.event.gesture.Manager,
/**
* Local alias for {@link ns.event}
* @property {Object}
* @member ns.event.gesture.Instance
* @private
* @static
*/
events = ns.event,
/**
* Alias for method {@link ns.util.object.merge}
* @property {Function} merge
* @member ns.event.gesture.Instance
* @private
* @static
*/
merge = ns.util.object.merge;
Gesture.Instance = function( element, options ) {
this.element = element;
this.eventDetectors = [];
this.options = merge({}, Gesture.defaults, options);
this.gestureManager = null;
this._init();
};
Gesture.Instance.prototype = {
/**
* Initialize gesture instance
* @method _init
* @member ns.event.gesture.Instance
* @protected
*/
_init: function() {
this.gestureManager = Manager.getInstance();
this.eventSender = merge({}, Detector.Sender, {
sendEvent: this.trigger.bind(this)
});
},
/**
* Find gesture detector
* @method _findGestureDetector
* @param {string} gesture gesture
* @member ns.event.gesture.Instance
* @protected
*/
_findGestureDetector: function( gesture ) {
var detectors = Detector.plugin,
detector, name;
for ( name in detectors ) {
if ( detectors.hasOwnProperty( name ) ) {
detector = detectors[ name ];
if ( detector.prototype.types.indexOf( gesture ) > -1 ) {
return detector;
}
}
}
},
/**
* Set options
* @method setOptions
* @param {Object} options options
* @chainable
* @member ns.event.gesture.Instance
*/
setOptions: function( options ) {
merge(this.options, options);
return this;
},
/**
* Add detector
* @method addDetector
* @param {Object} detectorStrategy strategy
* @chainable
* @member ns.event.gesture.Instance
*/
addDetector: function( detectorStrategy ) {
var detector = new Detector( detectorStrategy, this.eventSender ),
alreadyHasDetector = !!this.eventDetectors.length;
this.eventDetectors.push(detector);
if ( !!this.eventDetectors.length && !alreadyHasDetector ) {
this.gestureManager.register(this);
}
return this;
},
/**
* Remove detector
* @method removeDetector
* @param {Object} detectorStrategy strategy
* @chainable
* @member ns.event.gesture.Instance
*/
removeDetector: function( detectorStrategy ) {
var idx = this.eventDetectors.indexOf( detectorStrategy );
if ( idx > -1 ) {
this.eventDetectors.splice(idx, 1);
}
if ( !this.eventDetectors.length ) {
this.gestureManager.unregister(this);
}
return this;
},
/**
* Triggers the gesture event
* @method trigger
* @param {string} gesture gesture name
* @param {Object} eventInfo data provided to event object
* @member ns.event.gesture.Instance
*/
trigger: function( gesture, eventInfo ) {
return events.trigger(this.element, gesture, eventInfo, false);
},
/**
* Get HTML element assigned to gesture event instance
* @method getElement
* @member ns.event.gesture.Instance
*/
getElement: function() {
return this.element;
},
/**
* Get gesture event detectors assigned to instance
* @method getGestureDetectors
* @member ns.event.gesture.Instance
*/
getGestureDetectors: function() {
return this.eventDetectors;
},
/**
* Destroy instance
* @method destroy
* @member ns.event.gesture.Instance
*/
destroy: function( ) {
this.element = null;
this.eventHandlers = {};
this.gestureManager = null;
this.eventSender = null;
this.eventDetectors.length = 0;
}
};
} ( ns, window ) );
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* # Gesture Plugin: drag
* Plugin enables drag event.
*
* @class ns.event.gesture.Drag
*/
( function ( ns, window, undefined ) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.Drag
* @private
* @static
*/
var Gesture = ns.event.gesture,
/**
* Local alias for {@link ns.event.gesture.Detector}
* @property {Object}
* @member ns.event.gesture.Drag
* @private
* @static
*/
Detector = ns.event.gesture.Detector,
/**
* Alias for method {@link ns.util.object.merge}
* @property {Function} merge
* @member ns.event.gesture.Drag
* @private
* @static
*/
merge = ns.util.object.merge,
// TODO UA test will move to support.
tizenBrowser = !!window.navigator.userAgent.match(/tizen/i);
ns.event.gesture.Drag = Detector.plugin.create({
/**
* Gesture name
* @property {string} [name="drag"]
* @member ns.event.gesture.Drag
*/
name: "drag",
/**
* Gesture Index
* @property {number} [index=400]
* @member ns.event.gesture.Drag
*/
index: 500,
/**
* Array of posible drag events
* @property {string[]} types
* @member ns.event.gesture.Drag
*/
types: ["drag", "dragstart", "dragend", "dragcancel", "dragprepare"],
/**
* Default values for drag gesture
* @property {Object} defaults
* @property {boolean} [defaults.blockHorizontal=false]
* @property {boolean} [defaults.blockVertical=false]
* @property {number} [defaults.threshold=10]
* @property {number} [defaults.delay=0]
* @member ns.event.gesture.Drag
*/
defaults: {
blockHorizontal: false,
blockVertical: false,
threshold: 20,
delay: 0
},
/**
* Triggered
* @property {boolean} [triggerd=false]
* @member ns.event.gesture.Drag
*/
triggerd: false,
/**
* Handler for drag gesture
* @method handler
* @param {Event} gestureEvent gesture event
* @param {Object} sender event's sender
* @param {Object} options options
* @return {ns.event.gesture.Result.PENDING|ns.event.gesture.Result.END|ns.event.gesture.Result.FINISHED|ns.event.gesture.Result.BLOCK}
* @member ns.event.gesture.Drag
*/
handler: function( gestureEvent, sender, options ) {
var ge = gestureEvent,
threshold = options.threshold,
result = Gesture.Result.PENDING,
event = {
drag: this.types[0],
start: this.types[1],
end: this.types[2],
cancel: this.types[3],
prepare: this.types[4]
},
direction = ge.direction;
if ( !this.triggerd && ge.eventType === Gesture.Event.MOVE ) {
if ( Math.abs(ge.deltaX) < threshold && Math.abs(ge.deltaY) < threshold ) {
if ( !tizenBrowser ) {
ge.preventDefault();
}
return Gesture.Result.PENDING;
}
if ( options.delay && ge.deltaTime < options.delay ) {
if ( !tizenBrowser ) {
ge.preventDefault();
}
return Gesture.Result.PENDING;
}
if ( options.blockHorizontal && Gesture.utils.isHorizontal( ge.direction ) ||
options.blockVertical && Gesture.utils.isVertical( ge.direction ) ) {
return Gesture.Result.FINISHED;
}
this.fixedStartPointX = 0;
this.fixedStartPointY = 0;
if ( Gesture.utils.isHorizontal( ge.direction ) ) {
this.fixedStartPointX = ( ge.deltaX < 0 ? 1 : -1 ) * threshold;
} else {
this.fixedStartPointY = ( ge.deltaY < 0 ? 1 : -1 ) * threshold;
}
}
if ( options.blockHorizontal ) {
direction = ge.deltaY < 0 ? Gesture.Direction.UP : Gesture.Direction.DOWN;
}
if ( options.blockVertical ) {
direction = ge.deltaX < 0 ? Gesture.Direction.LEFT : Gesture.Direction.RIGHT;
}
ge = merge({}, ge, {
deltaX: ge.deltaX + this.fixedStartPointX,
deltaY: ge.deltaY + this.fixedStartPointY,
estimatedDeltaX: ge.estimatedDeltaX + this.fixedStartPointX,
estimatedDeltaY: ge.estimatedDeltaY + this.fixedStartPointY,
direction: direction
});
switch( ge.eventType ) {
case Gesture.Event.START:
this.triggerd = false;
if (sender.sendEvent( event.prepare, ge ) === false) {
result = Gesture.Result.FINISHED;
}
break;
case Gesture.Event.MOVE:
if ( !this.triggerd ) {
if (sender.sendEvent( event.start, ge ) === false) {
result = Gesture.Result.FINISHED;
ge.preventDefault();
break;
}
}
result = sender.sendEvent( event.drag, ge ) ? Gesture.Result.RUNNING : Gesture.Result.FINISHED;
ge.preventDefault();
this.triggerd = true;
break;
case Gesture.Event.BLOCKED:
case Gesture.Event.END:
result = Gesture.Result.FINISHED;
if ( this.triggerd ) {
sender.sendEvent( event.end, ge );
ge.preventDefault();
this.triggerd = false;
}
break;
case Gesture.Event.CANCEL:
result = Gesture.Result.FINISHED;
if ( this.triggerd ) {
sender.sendEvent( event.cancel, ge );
ge.preventDefault();
this.triggerd = false;
}
break;
}
return result;
}
});
} ( ns, window ) );
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Gesture Plugin: swipe
* Plugin enables swipe event.
*
* @class ns.event.gesture.Swipe
*/
( function ( ns, window, undefined ) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.Swipe
* @private
* @static
*/
var Gesture = ns.event.gesture,
/**
* Local alias for {@link ns.event.gesture.Detector}
* @property {Object}
* @member ns.event.gesture.Swipe
* @private
* @static
*/
Detector = ns.event.gesture.Detector;
ns.event.gesture.Swipe = Detector.plugin.create({
/**
* Gesture name
* @property {string} [name="swipe"]
* @member ns.event.gesture.Swipe
*/
name: "swipe",
/**
* Gesture Index
* @property {number} [index=400]
* @member ns.event.gesture.Swipe
*/
index: 400,
/**
* Default values for swipe gesture
* @property {Object} defaults
* @property {number} [defaults.timeThreshold=400]
* @property {number} [defaults.velocity=0.6]
* @property {ns.event.gesture.HORIZONTAL|ns.event.gesture.VERTICAL} [defaults.orientation=ns.event.gesture.HORIZONTAL]
* @member ns.event.gesture.Swipe
*/
defaults: {
timeThreshold: 400,
velocity: 0.6,
orientation: Gesture.Orientation.HORIZONTAL
},
/**
* Handler for swipe gesture
* @method handler
* @param {Event} gestureEvent gesture event
* @param {Object} sender event's sender
* @param {Object} options options
* @return {ns.event.gesture.Result.PENDING|ns.event.gesture.Result.END|ns.event.gesture.Result.FINISHED|ns.event.gesture.Result.BLOCK}
* @member ns.event.gesture.Swipe
*/
handler: function( gestureEvent, sender, options ) {
var ge = gestureEvent,
result = Gesture.Result.PENDING;
if ( ge.eventType !== Gesture.Event.END ) {
return result;
}
if ( ( ge.deltaTime > options.timeThreshold ) ||
( options.orientation !== Gesture.utils.getOrientation( ge.direction ) ) ) {
result = Gesture.Result.FINISHED;
return result;
}
if( ge.velocityX > options.velocity || ge.velocityY > options.velocity ) {
sender.sendEvent( this.name, gestureEvent );
result = Gesture.Result.FINISHED | Gesture.Result.BLOCK;
}
return result;
}
});
} ( ns, window ) );
/*global ns, window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* # Gesture Plugin: pinch
* Plugin enables pinch event.
*
* @class ns.event.gesture.Pinch
*/
( function ( ns, window, undefined ) {
"use strict";
/**
* Local alias for {@link ns.event.gesture}
* @property {Object}
* @member ns.event.gesture.Pinch
* @private
* @static
*/
var Gesture = ns.event.gesture,
/**
* Local alias for {@link ns.event.gesture.Detector}
* @property {Object}
* @member ns.event.gesture.Pinch
* @private
* @static
*/
Detector = ns.event.gesture.Detector;
ns.event.gesture.Pinch = Detector.plugin.create({
/**
* Gesture name
* @property {string} [name="pinch"]
* @member ns.event.gesture.Pinch
*/
name: "pinch",
/**
* Gesture Index
* @property {number} [index=300]
* @member ns.event.gesture.Pinch
*/
index: 300,
/**
* Array of posible pinch events
* @property {string[]} types
* @member ns.event.gesture.Pinch
*/
types: ["pinchstart", "pinchmove", "pinchend", "pinchcancel", "pinchin", "pinchout"],
/**
* Default values for pinch gesture
* @property {Object} defaults
* @property {number} [defaults.velocity=0.6]
* @property {number} [defaults.timeThreshold=400]
* @member ns.event.gesture.Pinch
*/
defaults: {
velocity: 0.6,
timeThreshold: 400
},
/**
* Triggered
* @property {boolean} [triggerd=false]
* @member ns.event.gesture.Pinch
*/
triggerd: false,
/**
* Handler for pinch gesture
* @method handler
* @param {Event} gestureEvent gesture event
* @param {Object} sender event's sender
* @param {Object} options options
* @return {ns.event.gesture.Result.PENDING|ns.event.gesture.Result.END|ns.event.gesture.Result.FINISHED|ns.event.gesture.Result.BLOCK}
* @member ns.event.gesture.Pinch
*/
handler: function ( gestureEvent, sender, options ) {
var ge = gestureEvent,
result = Gesture.Result.PENDING,
event = {
start: this.types[0],
move: this.types[1],
end: this.types[2],
cancel: this.types[3],
in: this.types[4],
out: this.types[5]
};
switch( ge.eventType ) {
case Gesture.Event.MOVE:
if (ge.pointers.length === 1 && ge.distance > 35) {
result = Gesture.Result.FINISHED;
return result;
} else if ( !this.triggerd && ge.pointers.length >= 2) {
this.triggerd = true;
sender.sendEvent( event.start, ge );
ge.preventDefault();
result = Gesture.Result.RUNNING;
} else if ( this.triggerd) {
if ( ( ge.deltaTime < options.timeThreshold ) &&
( ge.velocityX > options.velocity || ge.velocityY > options.velocity ) ) {
if (ge.scale < 1) {
sender.sendEvent( event.in, gestureEvent );
} else {
sender.sendEvent( event.out, gestureEvent );
}
ge.preventDefault();
this.triggerd = false;
result = Gesture.Result.FINISHED | Gesture.Result.BLOCK;
return result;
} else {
sender.sendEvent( event.move, ge );
ge.preventDefault();
result = Gesture.Result.RUNNING;
}
}
break;
case Gesture.Event.BLOCKED:
case Gesture.Event.END:
if ( this.triggerd ) {
sender.sendEvent( event.end, ge );
ge.preventDefault();
this.triggerd = false;
result = Gesture.Result.FINISHED;
}
break;
case Gesture.Event.CANCEL:
if ( this.triggerd ) {
sender.sendEvent( event.cancel, ge );
ge.preventDefault();
this.triggerd = false;
result = Gesture.Result.FINISHED;
}
break;
}
return result;
}
});
} ( ns, window ) );
/*global window, define, CustomEvent */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @class ns.event.gesture
*/
(function (ns) {
"use strict";
var instances = [],
gesture = ns.event.gesture || {};
/**
* Find instance by element
* @method findInstance
* @param {HTMLElement} element
* @return {ns.event.gesture.Instance}
* @member ns.event
* @static
* @private
*/
function findInstance(element) {
var instance;
instances.forEach(function(item) {
if (item.element === element) {
instance = item.instance;
}
});
return instance;
}
/**
* Remove instance from instances by element
* @method removeInstance
* @param {HTMLElement} element
* @member ns.event
* @static
* @private
*/
function removeInstance(element) {
instances.forEach(function(item, key) {
if (item.element === element) {
instances.splice(key, 1);
}
});
}
/**
* Enable gesture handlingo on given HTML element or object
* @method enableGesture
* @param {HTMLElement} element
* @param {...Object} [gesture] Gesture object {@link ns.event.gesture}
* @member ns.event
*/
ns.event.enableGesture = function() {
var element = arguments[0],
gestureInstance = findInstance( element ),
length = arguments.length,
i = 1;
if ( !gestureInstance ) {
gestureInstance = new gesture.Instance(element);
instances.push({element: element, instance: gestureInstance});
}
for ( ; i < length; i++ ) {
gestureInstance.addDetector( arguments[i] );
}
};
/**
* Disable gesture handling from given HTML element or object
* @method disableGesture
* @param {HTMLElement} element
* @param {...Object} [gesture] Gesture object {@link ns.event.gesture}
* @member ns.event
*/
ns.event.disableGesture = function() {
var element = arguments[0],
gestureInstance = findInstance( element ),
length = arguments.length,
i = 1;
if ( !gestureInstance ) {
return;
}
if ( length > 1 ) {
gestureInstance.removeDetector( arguments[i] );
} else {
gestureInstance.destroy();
removeInstance( element );
}
};
ns.event.gesture = gesture;
}(ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
*/
(function (ns) {
"use strict";
ns.widget.core.viewswitcher = ns.widget.core.viewswitcher || {};
}(ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* #type namespace
* ViewSwitcher animation
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
* @class ns.widget.core.ViewSwitcher.animation
*/
(function (window, ns) {
"use strict";
/** @namespace ns.widget.wearable */
ns.widget.core.viewswitcher.animation = ns.widget.core.viewswitcher.animation || {};
}(window, ns));
/*global window, define, Event, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #Animation Interface
* Interface for animation for used viewswitcher
* @class ns.widget.core.viewswitcher.animation.interface
*/
(function (document, ns) {
"use strict";
ns.widget.core.viewswitcher.animation.interface = {
/**
* Init views position
* @method initPosition
* @param views array
* @param active index
* @static
* @member ns.widget.core.viewswitcher.animation.interface
*/
initPosition: function (/* views array, active index */) {
},
/**
* Animate views
* @method animate
* @param views array
* @param active index
* @param position
* @static
* @member ns.widget.core.viewswitcher.animation.interface
*/
animate: function (/* views array, active index, position */) {
},
/**
* Reset views position
* @method resetPosition
* @param views array
* @param active index
* @static
* @member ns.widget.core.viewswitcher.animation.interface
*/
resetPosition: function (/* views array, active index */) {
}
};
}(window.document, ns));
/*global window, define, Event, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #Animation carousel
*
* carousel is animation type of ViewSwitcher
*
* @class ns.widet.core.ViewSwitcher.animation.carousel
* @extends ns.widget.core.ViewSwitcher.animation.interface
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
*/
(function (document, ns) {
"use strict";
var object = ns.util.object,
utilDOM = ns.util.DOM,
animation = ns.widget.core.viewswitcher.animation,
animationInterface = animation.interface,
DEFAULT = {
PERSPECTIVE: 280,
ZINDEX_TOP: 3,
ZINDEX_MIDDLE: 2,
ZINDEX_BOTTOM: 1,
DIM_LEVEL: 6
},
options = {
useDim: true,
dimLevel: DEFAULT.DIM_LEVEL
},
classes = {
CAROUSEL: "ui-view-carousel",
CAROUSEL_ACTIVE: "ui-view-carousel-active",
CAROUSEL_LEFT: "ui-view-carousel-left",
CAROUSEL_RIGHT: "ui-view-carousel-right",
CAROUSEL_DIM: "ui-view-carousel-dim"
};
function translate(element, x, y, z, duration) {
if (duration) {
utilDOM.setPrefixedStyle(element, "transition", utilDOM.getPrefixedValue("transform " + duration / 1000 + "s ease-out"));
}
utilDOM.setPrefixedStyle(element, "transform", "translate3d(" + x + "px, " + y + "px, " + z + "px)");
}
function resetStyle(element) {
element.style.left = "";
element.style.right = "";
element.style.zIndex = DEFAULT.ZINDEX_MIDDLE;
element.style.transform = "translateZ(" + -element.parentNode.offsetWidth / 2 + "px)";
element.style.webkitTransform = "translateZ(" + -element.parentNode.offsetWidth / 2 + "px)";
}
animation.carousel = object.merge({}, animationInterface, {
/**
* Init views position
* @method initPosition
* @param views array
* @param active index
* @static
* @member ns.widget.core.ViewSwitcher.animation.interface
*/
initPosition: function (views, index) {
var viewSwitcher = views[0].parentNode,
vsOffsetWidth = viewSwitcher.offsetWidth,
dimElement,
i, len;
viewSwitcher.classList.add(classes.CAROUSEL);
viewSwitcher.style.webkitPerspective = DEFAULT.PERSPECTIVE;
if (options.useDim) {
len = views.length;
for (i = 0; i < len; i++) {
dimElement = document.createElement("DIV");
dimElement.classList.add(classes.CAROUSEL_DIM);
views[i].appendChild(dimElement);
}
}
views[index].classList.add(classes.CAROUSEL_ACTIVE);
if (index > 0) {
views[index - 1].classList.add(classes.CAROUSEL_LEFT);
views[index - 1].style.transform = "translateZ(" + -vsOffsetWidth / 2 + "px)";
}
if (index < views.length - 1) {
views[index + 1].classList.add(classes.CAROUSEL_RIGHT);
views[index + 1].style.transform = "translateZ(" + -vsOffsetWidth / 2 + "px)";
}
},
/**
* Animate views
* @method animate
* @param views array
* @param active index
* @param position [0 - 100 or -100 - 0]
* @static
* @member ns.widget.core.ViewSwitcher.animation.interface
*/
animate: function (views, index, position) {
var viewSwitcher = views[0].parentNode,
vsWidth = viewSwitcher.offsetWidth,
vsHalfWidth = vsWidth / 2,
left = index > 0 ? views[index - 1] : undefined,
right = index < views.length - 1 ? views[index + 1] : undefined,
active = views[index],
ex = position / 100 * vsWidth,
halfEx = ex / 2,
centerPosition = (vsHalfWidth - active.offsetWidth / 2),
adjPosition = (centerPosition/ (vsHalfWidth * 0.6)),
absEx = Math.abs(ex),
absPosition = Math.abs(position),
mark = position < 0 ? 1 : -1,
edge = vsHalfWidth * 0.2 * mark,
// edgeDeltaX -> -mark * (2 * (0.8 * vsHalfWidth)) - halfEx
edgeDeltaX = -mark * 1.6 * vsHalfWidth - halfEx,
minusDeltaX = -vsHalfWidth - halfEx,
plusDeltaX = -vsHalfWidth + halfEx,
hidingDeltaX = -halfEx * 0.2,
prev, next, beforePrev, afterNext;
active.style.left = (vsWidth - active.offsetWidth) / 2 + "px";
active.style.zIndex = DEFAULT.ZINDEX_TOP;
next = ex < 0 ? right : left;
afterNext = ex < 0 ? (next && next.nextElementSibling) : (next && next.previousElementSibling);
prev = ex < 0 ? left : right;
beforePrev = ex < 0 ? (prev && prev.previousElementSibling) : (prev && prev.nextElementSibling);
if (next) {
if (absEx < vsWidth * 0.2) {
next.style.zIndex = DEFAULT.ZINDEX_MIDDLE;
translate(next, -halfEx * adjPosition, 0, ex < 0 ? minusDeltaX : plusDeltaX);
} else {
active.style.zIndex = DEFAULT.ZINDEX_MIDDLE;
next.style.zIndex = DEFAULT.ZINDEX_TOP;
translate(next, (2 * edge + halfEx) * adjPosition, 0, ex < 0 ? minusDeltaX : plusDeltaX);
}
if (afterNext) {
afterNext.classList.add(ex < 0 ? classes.CAROUSEL_RIGHT : classes.CAROUSEL_LEFT);
translate(afterNext, (ex < 0 ? minusDeltaX : -plusDeltaX) * 0.6, 0, -vsWidth - halfEx * mark);
}
}
if (prev) {
if (beforePrev) {
beforePrev.classList.remove(ex < 0 ? classes.CAROUSEL_LEFT : classes.CAROUSEL_RIGHT);
}
prev.style.zIndex = DEFAULT.ZINDEX_BOTTOM;
translate(prev, hidingDeltaX, 0, ex < 0 ? plusDeltaX : minusDeltaX);
}
if (absEx < vsWidth * 0.8) {
translate(active, halfEx * adjPosition, 0, halfEx * mark);
} else {
translate(active, edgeDeltaX * adjPosition, 0, halfEx * mark);
}
if (options.useDim) {
active.querySelector("." + classes.CAROUSEL_DIM).style.opacity = absPosition * options.dimLevel / 1000;
if (next) {
next.querySelector("." + classes.CAROUSEL_DIM).style.opacity = options.dimLevel / 10 * (1 - absPosition / 100);
}
}
},
/**
* Reset views position
* @method resetPosition
* @param views array
* @param active index
* @static
* @member ns.widget.core.ViewSwitcher.animation.interface
*/
resetPosition: function (views, index) {
var viewSwitcher = views[0].parentNode,
active = views[index],
rightElements = viewSwitcher.querySelectorAll("." + classes.CAROUSEL_RIGHT),
leftElements = viewSwitcher.querySelectorAll("." + classes.CAROUSEL_LEFT),
i, len;
viewSwitcher.querySelector("." + classes.CAROUSEL_ACTIVE).classList.remove(classes.CAROUSEL_ACTIVE);
active.classList.add(classes.CAROUSEL_ACTIVE);
active.style.transform = "";
active.style.webkitTransform = "";
len = rightElements.length;
for (i = 0; i < len; i++) {
rightElements[i].classList.remove(classes.CAROUSEL_RIGHT);
}
if (index < views.length - 1) {
views[index + 1].classList.add(classes.CAROUSEL_RIGHT);
resetStyle(views[index + 1]);
}
len = leftElements.length;
for (i = 0; i < len; i++) {
leftElements[i].classList.remove(classes.CAROUSEL_LEFT);
}
if (index > 0) {
views[index - 1].classList.add(classes.CAROUSEL_LEFT);
resetStyle(views[index - 1]);
}
}
});
animation.carousel.options = options;
}(window.document, ns));
/*global window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #ViewSwitcher Component
* ViewSwitcher component is controller for each view elements is changing position.
* This component managed to animation, views position, events and get/set active view index.
* If you want to change the view as various animating, you should wrap views as the ViewSwitcher element then
* ViewSwitcher would set views position and start to manage to gesture event.
*
* ##Set and Get the active index
* You can set or get the active index as the setActiveIndex() and getActiveIndex()
*
* @class ns.widget.core.viewswitcher.ViewSwitcher
* @extends ns.widget.BaseWidget
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
*/
(function (document, ns) {
"use strict";
/**
* @property {Object} Widget Alias for {@link ns.widget.BaseWidget}
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @private
* @static
*/
var BaseWidget = ns.widget.BaseWidget,
events = ns.event,
engine = ns.engine,
utilsObject = ns.util.object,
Gesture = ns.event.gesture,
/**
* Default values
*/
DEFAULT = {
ACTIVE_INDEX: 0,
ANIMATION_TYPE: "carousel",
ANIMATION_SPEED: 30,
ANIMATION_TIMING_FUNCTION: "ease-out"
},
/**
* ViewSwitcher triggered some customEvents
* viewchangestart : This event has been triggerred when view changing started.
* viewchangeend : This event has been triggerred when view changing ended.
* viewchange: This event has been triggerred when view changing complete to user.
*/
EVENT_TYPE = {
CHANGE_START: "viewchangestart",
CHANGE_END: "viewchangeend",
CHANGE: "viewchange"
},
/**
* ViewSwitcher constructor
* @method ViewSwitcher
*/
ViewSwitcher = function () {
var self = this;
self.options = {};
self._ui = {};
},
/**
* Dictionary object containing commonly used widget classes
* @property {Object} classes
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @private
* @static
* @readonly
*/
classes = {
VIEW: "ui-view",
VIEW_ACTIVE: "ui-view-active",
ANIMATION_TYPE: "ui-animation-"
},
/**
* {Object} ViewSwitcher widget prototype
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @private
* @static
*/
prototype = new BaseWidget();
ViewSwitcher.prototype = prototype;
ViewSwitcher.classes = classes;
/**
* Configure of ViewSwitcher component
* @method _configure
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._configure = function () {
var self = this;
/**
* ViewSwitcher containing some options
* @property {number} ViewSwitcher default active index (Default is 0)
* @property {string} ViewSwitcher animation type (Default is "carousel")
* @property {number} ViewSwitcher animation speed (Default is 18)
*/
self.options = utilsObject.merge(self.options, {
active: DEFAULT.ACTIVE_INDEX,
animationType: DEFAULT.ANIMATION_TYPE,
animationSpeed: DEFAULT.ANIMATION_SPEED
});
};
/**
* Build structure of ViewSwitcher component
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement} Returns built element
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._build = function (element) {
var self = this,
ui = self._ui;
ui._views = element.querySelectorAll("." + classes.VIEW);
return element;
};
/**
* Initialization of ViewSwitcher component
* @method _init
* @param {HTMLElement} element
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._init = function (element) {
var self = this;
self._elementOffsetWidth = element.offsetWidth;
self._initPosition();
return element;
};
/**
* Init position of Views inner ViewSwitcher
* @method _initPosition
* @param {HTMLElement} element
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._initPosition = function () {
var self = this,
views = self._ui._views,
options = self.options,
activeIndex = self._getActiveIndex();
self._type = ns.widget.core.viewswitcher.animation[options.animationType];
self._type.initPosition(views, activeIndex);
self._activeIndex = activeIndex;
};
/**
* Get the active index as view has the "ui-view-active" or not
* @method _getActiveIndex
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._getActiveIndex = function() {
var self = this,
ui = self._ui,
views = ui._views,
i, len;
len = views.length;
for (i = 0; i < len; i++) {
if (views[i].classList.contains(classes.VIEW_ACTIVE)) {
return i;
}
}
return self.options.active;
};
/**
* Binds events to a ViewSwitcher component
* @method _bindEvents
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._bindEvents = function() {
var self = this,
element = self.element;
events.enableGesture(
element,
new events.gesture.Drag({
blockVertical: true,
threshold: 0
}),
new events.gesture.Swipe({
orientation: Gesture.Orientation.HORIZONTAL
})
);
events.on(element, "drag dragstart dragend swipe", self, false);
};
/**
* Handle events
* @method handleEvent
* @param {Event} event
* @member ns.widget.core.viewswitcher.ViewSwitcher
*/
prototype.handleEvent = function(event) {
var self = this;
switch (event.type) {
case "drag":
self._onDrag(event);
break;
case "dragstart":
self._onDragStart(event);
break;
case "dragend":
case "swipe":
self._onDragEnd(event);
break;
}
};
/**
* Drag event handler
* @method _onDrag
* @param {Event} event
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._onDrag = function(event) {
var self = this,
direction = event.detail.direction,
ex = event.detail.estimatedDeltaX,
deltaX = ex / self._elementOffsetWidth * 100,
ui = self._ui,
active = ui._views[self._activeIndex];
if ((direction === "left" && !active.nextElementSibling) || (direction === "right" && !active.previousElementSibling)) {
return;
}
if (self._dragging && !self._isAnimating && Math.abs(deltaX) < 100) {
self._type.animate(ui._views, self._activeIndex, deltaX);
self._triggerChange(deltaX);
}
};
/**
* DragStart event handler
* @method _onDragStart
* @param {Event} event
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._onDragStart = function(event) {
var self = this,
direction = event.detail.direction,
ui = self._ui,
active = ui._views[self._activeIndex];
if ((direction === "left" && !active.nextElementSibling) || (direction === "right" && !active.previousElementSibling) || self._dragging) {
return;
}
self._dragging = true;
};
/**
* DragEnd event handler
* @method _onDragEnd
* @param {Event} event
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._onDragEnd = function(event) {
var self = this,
ui = self._ui,
active = ui._views[self._activeIndex],
direction = event.detail.direction,
estimatedDeltaX = event.detail.estimatedDeltaX;
if (!self._dragging || self._isAnimating
|| (direction === "left" && !active.nextElementSibling) || (direction === "right" && !active.previousElementSibling)) {
return;
}
self._lastDirection = direction;
if (event.type === "dragend" && Math.abs(estimatedDeltaX) < self._elementOffsetWidth / 2) {
direction = "backward";
}
self.trigger(EVENT_TYPE.CHANGE_START);
self._requestFrame(estimatedDeltaX, direction);
};
prototype._triggerChange = function(estimatedDeltaX) {
var self = this,
absEx = Math.abs(estimatedDeltaX);
if (absEx > 50 && !self._changed) {
self.trigger(EVENT_TYPE.CHANGE, {
index: self._activeIndex + (estimatedDeltaX < 0 ? 1 : -1)
});
self._changed = true;
} else if (absEx < 50 && self._changed){
self.trigger(EVENT_TYPE.CHANGE, {
index: self._activeIndex
});
self._changed = false;
}
};
/**
* Animate views as the requestAnimationFrame.
* @method _requestFrame
* @param {string} animation direction
* @param {string} animation timing type (ease-out|linear)
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._requestFrame = function(estimatedDeltaX, direction, animationTiming) {
var self = this,
elementOffsetWidth = self._elementOffsetWidth,
animationTimingFunction = animationTiming ? animationTiming : DEFAULT.ANIMATION_TIMING_FUNCTION,
isStop = false,
lastDirection = self._lastDirection,
ui = self._ui,
ex = estimatedDeltaX,
deltaX = ex / elementOffsetWidth * 100,
animationFrame,
validDirection,
stopPosition,
mark;
if (direction === "backward") {
validDirection = lastDirection === "left" ? "right" : "left";
if (lastDirection === "left" && ex > 0
|| lastDirection === "right" && ex < 0) {
isStop = true;
stopPosition = 0;
}
} else {
validDirection = direction;
if (Math.abs(ex) > elementOffsetWidth) {
isStop = true;
stopPosition = 100;
}
}
mark = validDirection === "left" ? -1 : 1;
if (isStop) {
self._type.animate(ui._views, self._activeIndex, stopPosition * mark);
webkitCancelRequestAnimationFrame(animationFrame);
if (direction !== "backward") {
ui._views[self._activeIndex].classList.remove(classes.VIEW_ACTIVE);
self._activeIndex = self._activeIndex - mark;
self._type.resetPosition(ui._views, self._activeIndex);
ui._views[self._activeIndex].classList.add(classes.VIEW_ACTIVE);
}
self._dragging = false;
self._isAnimating = false;
self._changed = false;
self.trigger(EVENT_TYPE.CHANGE_END);
return;
}
self._type.animate(ui._views, self._activeIndex, deltaX);
self._triggerChange(deltaX);
self._isAnimating = true;
if (animationTimingFunction === "ease-out") {
if (Math.abs(ex) > elementOffsetWidth * 0.95) {
ex = ex + mark;
} else {
ex = ex + self.options.animationSpeed * mark;
}
} else if (animationTimingFunction === "linear") {
ex = ex + self.options.animationSpeed * mark;
}
animationFrame = webkitRequestAnimationFrame(self._requestFrame.bind(self, ex, direction, animationTiming));
};
/**
* Set the active view
* @method setActiveIndex
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @public
*/
prototype.setActiveIndex = function(index) {
var self = this,
latestActiveIndex = self._activeIndex,
interval = latestActiveIndex - index,
direction, i, len;
if (!self._isAnimating && index < self._ui._views.length && index >= 0) {
self._lastDeltaX = 0;
if (interval < 0) {
direction = "left";
} else {
direction = "right";
}
len = Math.abs(interval);
self._lastDirection = direction;
for (i = 0; i < len; i++) {
self.trigger(EVENT_TYPE.CHANGE_START);
self._requestFrame(0, direction, "linear");
}
}
};
/**
* Get the active view index
* @method getActiveIndex
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @public
*/
prototype.getActiveIndex = function() {
return this._activeIndex;
};
/**
* Destroys ViewSwitcher widget
* @method _destroy
* @member ns.widget.core.viewswitcher.ViewSwitcher
* @protected
*/
prototype._destroy = function() {
var element = this.element;
events.disableGesture(element);
events.off(element, "drag dragstart dragend", this, false);
this.options = null;
this._ui = null;
};
ns.widget.core.viewswitcher.ViewSwitcher = ViewSwitcher;
engine.defineWidget(
"ViewSwitcher",
"[data-role='viewSwitcher'], .ui-view-switcher",
[
"setActiveIndex",
"getActiveIndex"
],
ViewSwitcher
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #PageIndicator Widget
* Widget create dots page indicator.
* @class ns.widget.core.PageIndicator
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
PageIndicator = function () {
var self = this;
self._activeIndex = null;
self.options = {};
},
classes = {
indicator: "ui-page-indicator",
indicatorActive: "ui-page-indicator-active",
indicatorItem: "ui-page-indicator-item",
linearIndicator: "ui-page-indicator-linear",
circularIndicator: "ui-page-indicator-circular"
},
maxDots = {
IN_CIRCLE: 60,
IN_LINEAR: 5
},
layoutType = {
LINEAR: "linear",
CIRCULAR: "circular"
},
DISTANCE_FROM_EDGE = 15,
prototype = new BaseWidget();
PageIndicator.classes = classes;
prototype._configure = function () {
/**
* Options for widget.
* @property {Object} options
* @property {number} [options.maxPage=null] Maximum number of dots(pages) in indicator.
* @property {number} [options.numberOfPages=null] Number of pages to be linked to PageIndicator.
* @property {string} [options.layout="linear"] Layout type of page indicator.
* @property {number} [options.intervalAngle=6] angle between each dot in page indicator.
* @member ns.widget.core.PageIndicator
*/
this.options = {
maxPage: null,
numberOfPages: null,
layout: "linear",
intervalAngle: 6
};
};
/**
* Build PageIndicator
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._build = function (element) {
var self = this;
self._createIndicator(element);
if (self.options.layout === layoutType.CIRCULAR) {
self._circularPositioning(element);
}
return element;
};
/**
* Create HTML elements for PageIndicator
* @method _createIndicator
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._createIndicator = function (element) {
var self = this,
i,
len,
maxPage,
span,
numberOfPages = self.options.numberOfPages;
if(numberOfPages === null) {
ns.error("build error: numberOfPages is null");
return;
}
self.options.layout = self.options.layout.toLowerCase();
if (self.options.layout === layoutType.CIRCULAR) {
element.classList.remove(classes.linearIndicator);
element.classList.add(classes.circularIndicator);
} else {
element.classList.remove(classes.circularIndicator);
element.classList.add(classes.linearIndicator);
}
maxPage = self._getMaxPage();
len = numberOfPages < maxPage ? numberOfPages : maxPage;
for(i = 0; i < len; i++) {
span = document.createElement("span");
span.classList.add(classes.indicatorItem);
element.appendChild(span);
}
};
/**
* Make circular positioned indicator
* @method _circularPositioning
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._circularPositioning = function (element) {
var self = this,
items = element.children,
numberOfDots = items.length,
intervalAngle = self.options.intervalAngle - "0",
translatePixel,
style,
i;
translatePixel = element.offsetWidth / 2 - DISTANCE_FROM_EDGE;
for(i=0;i<numberOfDots;i++) {
style = "rotate(" + (i * intervalAngle - 90 - (numberOfDots-1) * intervalAngle * 0.5) + "deg) translate(" +
translatePixel + "px) ";
items[i].style.transform = style;
}
};
/**
* Return maximum number of dots(pages) in indicator
* @method _getMaxPage
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._getMaxPage = function() {
var self = this,
options = self.options,
maxPage;
if (options.layout === layoutType.CIRCULAR) {
maxPage = options.maxPage || maxDots.IN_CIRCLE;
} else {
maxPage = options.maxPage || maxDots.IN_LINEAR;
}
return maxPage;
};
/**
* Remove contents of HTML elements for PageIndicator
* @method _removeIndicator
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._removeIndicator = function (element) {
element.textContent = "";
};
/**
* This method sets a dot to active state.
* @method setActive
* @param {number} position index to be active state.
* @member ns.widget.core.PageIndicator
*/
prototype.setActive = function (position) {
var self = this,
dotIndex = position,
elPageIndicatorItems = self.element.children,
maxPage,
numberOfPages = self.options.numberOfPages,
middle,
numberOfCentralDotPages = 0,
indicatorActive = classes.indicatorActive,
previousActive;
if(position === null || position === undefined) {
return;
}
self._activeIndex = position;
maxPage = self._getMaxPage();
middle = window.parseInt(maxPage/2, 10);
if(numberOfPages > maxPage) {
numberOfCentralDotPages = numberOfPages - maxPage;
} else if(numberOfPages === null) {
ns.error("setActive error: numberOfPages is null");
return;
} else if(numberOfPages === 0) {
return;
}
previousActive = self.element.querySelector("." + indicatorActive);
if(previousActive) {
previousActive.classList.remove(indicatorActive);
}
if ((middle < position) && (position <= (middle + numberOfCentralDotPages))) {
dotIndex = middle;
} else if (position > (middle + numberOfCentralDotPages)) {
dotIndex = position - numberOfCentralDotPages;
}
elPageIndicatorItems[dotIndex].classList.add(indicatorActive);
};
/**
* Refresh widget structure
* @method _refresh
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._refresh = function () {
var self = this,
element = self.element;
self._removeIndicator(element);
self._createIndicator(element);
if (self.options.layout === layoutType.CIRCULAR) {
self._circularPositioning(element);
}
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.core.PageIndicator
*/
prototype._destroy = function () {
this._removeIndicator(this.element);
};
PageIndicator.prototype = prototype;
ns.widget.core.PageIndicator = PageIndicator;
engine.defineWidget(
"PageIndicator",
"[data-role='page-indicator'], .ui-page-indicator",
["setActive"],
PageIndicator,
"core"
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Scroller namespace
* Namespace contains classes and objects connected with scroller widget.
* @class ns.widget.wearable.scroller
* @author Maciej Urbanski <m.urbanski@samsung.com>
*/
(function (window, ns) {
"use strict";
ns.widget.core.scroller = ns.widget.core.scroller || {};
}(window, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* #Effect namespace
* Namespace with effects for scroller widget.
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @class ns.widget.core.scroller.effect
*/
(function (window, ns) {
"use strict";
ns.widget.core.scroller.effect = ns.widget.core.scroller.effect || {};
}(window, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # Bouncing effect
* Bouncing effect for scroller widget.
* @class ns.widget.core.scroller.effect.Bouncing
* @since 2.3
*/
(function (document, ns) {
"use strict";
// scroller.start event trigger when user try to move scroller
var utilsObject = ns.util.object,
selectors = ns.util.selectors,
Bouncing = function (scrollerElement, options) {
var self = this;
self._orientation = null;
self._maxValue = null;
self._container = null;
self._minEffectElement = null;
self._maxEffectElement = null;
self.options = utilsObject.merge({}, Bouncing.defaults, {scrollEndEffectArea: ns.getConfig("scrollEndEffectArea", Bouncing.defaults.scrollEndEffectArea)});
/**
* target element for bouncing effect
* @property {HTMLElement} targetElement
* @member ns.widget.core.scroller.effect.Bouncing
*/
self._targetElement = null;
self._isShow = false;
self._isDrag = false;
self._isShowAnimating = false;
self._isHideAnimating = false;
self._create(scrollerElement, options);
},
endEffectAreaType = {
content: "content",
screen: "screen"
},
defaults = {
duration: 500,
scrollEndEffectArea : "content"
},
classes = {
bouncingEffect: "ui-scrollbar-bouncing-effect",
page: "ui-page",
left: "ui-left",
right: "ui-right",
top: "ui-top",
bottom: "ui-bottom",
hide: "ui-hide",
show: "ui-show"
};
Bouncing.defaults = defaults;
Bouncing.prototype = {
_create: function (scrollerElement, options) {
var self = this;
if( self.options.scrollEndEffectArea === endEffectAreaType.content ){
self._container = scrollerElement;
} else {
self._container = selectors.getClosestByClass(scrollerElement, classes.page);
}
self._orientation = options.orientation;
self._maxValue = self._getValue( options.maxScrollX, options.maxScrollY );
self._initLayout();
},
_initLayout: function() {
var self = this,
minElement = self._minEffectElement = document.createElement("DIV"),
maxElement = self._maxEffectElement = document.createElement("DIV"),
className = classes.bouncingEffect;
if ( self._orientation === ns.widget.core.scroller.Scroller.Orientation.HORIZONTAL ) {
minElement.className = className + " " + classes.left;
maxElement.className = className + " " + classes.right;
} else {
minElement.className = className + " " + classes.top;
maxElement.className = className + " " + classes.bottom;
}
self._container.appendChild( minElement );
self._container.appendChild( maxElement );
minElement.addEventListener("animationEnd", this);
minElement.addEventListener("webkitAnimationEnd", this);
minElement.addEventListener("mozAnimationEnd", this);
minElement.addEventListener("msAnimationEnd", this);
minElement.addEventListener("oAnimationEnd", this);
maxElement.addEventListener("animationEnd", this);
maxElement.addEventListener("webkitAnimationEnd", this);
maxElement.addEventListener("mozAnimationEnd", this);
maxElement.addEventListener("msAnimationEnd", this);
maxElement.addEventListener("oAnimationEnd", this);
},
/**
* ...
* @method drag
* @param x
* @param y
* @member ns.widget.core.scroller.effect.Bouncing
*/
drag: function( x, y ) {
this._isDrag = true;
this._checkAndShow( x, y );
},
/**
* ...
* @method dragEnd
* @member ns.widget.core.scroller.effect.Bouncing
*/
dragEnd: function() {
var self = this;
if ( self._isShow && !self._isShowAnimating && !self._isHideAnimating ) {
self._beginHide();
}
self._isDrag = false;
},
/**
* Shows effect.
* @method show
* @member ns.widget.core.scroller.effect.Bouncing
*/
show: function() {
var self = this;
if ( self._targetElement ) {
self._isShow = true;
self._beginShow();
}
},
/**
* Hides effect.
* @method hide
* @member ns.widget.core.scroller.effect.Bouncing
*/
hide: function() {
var self = this;
if ( self._isShow ) {
self._minEffectElement.style.display = "none";
self._maxEffectElement.style.display = "none";
self._targetElement.classList.remove(classes.hide);
self._targetElement.classList.remove(classes.show);
}
self._isShow = false;
self._isShowAnimating = false;
self._isHideAnimating = false;
self._targetElement = null;
},
_checkAndShow: function( x, y ) {
var self = this,
val = self._getValue(x, y);
if ( !self._isShow ) {
if ( val >= 0 ) {
self._targetElement = self._minEffectElement;
self.show();
} else if ( val <= self._maxValue ) {
self._targetElement = self._maxEffectElement;
self.show();
}
} else if ( self._isShow && !self._isDrag && !self._isShowAnimating && !self._isHideAnimating ) {
self._beginHide();
}
},
_getValue: function(x, y) {
return this._orientation === ns.widget.core.scroller.Scroller.Orientation.HORIZONTAL ? x : y;
},
_beginShow: function() {
var self = this;
if ( !self._targetElement || self._isShowAnimating ) {
return;
}
self._targetElement.style.display = "block";
self._targetElement.classList.remove(classes.hide);
self._targetElement.classList.add(classes.show);
self._isShowAnimating = true;
self._isHideAnimating = false;
},
_finishShow: function() {
var self = this;
self._isShowAnimating = false;
if ( !self._isDrag ) {
self._targetElement.classList.remove(classes.show);
self._beginHide();
}
},
_beginHide: function() {
var self = this;
if ( self._isHideAnimating ) {
return;
}
self._targetElement.classList.remove(classes.show);
self._targetElement.classList.add(classes.hide);
self._isHideAnimating = true;
self._isShowAnimating = false;
},
_finishHide: function() {
var self = this;
self._isHideAnimating = false;
self._targetElement.classList.remove(classes.hide);
self.hide();
self._checkAndShow();
},
/**
* Supports events.
* @method handleEvent
* @member ns.widget.core.scroller.effect.Bouncing
*/
handleEvent: function( event ) {
if (event.type.toLowerCase().indexOf("animationend") > -1) {
if ( this._isShowAnimating ) {
this._finishShow();
} else if ( this._isHideAnimating ) {
this._finishHide();
}
}
},
/**
* Destroys effect.
* @method destroy
* @member ns.widget.core.scroller.effect.Bouncing
*/
destroy: function() {
var self = this,
maxEffectElement = this._maxEffectElement,
minEffectElement = this._minEffectElement;
minEffectElement.removeEventListener("animationEnd", this);
minEffectElement.removeEventListener("webkitAnimationEnd", this);
minEffectElement.removeEventListener("mozAnimationEnd", this);
minEffectElement.removeEventListener("msAnimationEnd", this);
minEffectElement.removeEventListener("oAnimationEnd", this);
maxEffectElement.removeEventListener("animationEnd", this);
maxEffectElement.removeEventListener("webkitAnimationEnd", this);
maxEffectElement.removeEventListener("mozAnimationEnd", this);
maxEffectElement.removeEventListener("msAnimationEnd", this);
maxEffectElement.removeEventListener("oAnimationEnd", this);
self._container.removeChild( minEffectElement );
self._container.removeChild( maxEffectElement );
self._container = null;
self._minEffectElement = null;
self._maxEffectElement = null;
self._targetElement = null;
self._isShow = null;
self._orientation = null;
self._maxValue = null;
}
};
ns.widget.core.scroller.effect.Bouncing = Bouncing;
}(window.document, ns));
/*global window, define, Event, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # Scroller Widget
* Widget creates scroller on content.
* @class ns.widget.core.scroller.Scroller
* @since 2.3
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
// scroller.start event trigger when user try to move scroller
var BaseWidget = ns.widget.BaseWidget,
Gesture = ns.event.gesture,
engine = ns.engine,
utilsObject = ns.util.object,
utilsEvents = ns.event,
eventTrigger = utilsEvents.trigger,
prototype = new BaseWidget(),
EffectBouncing = ns.widget.core.scroller.effect.Bouncing,
eventType = {
/**
* event trigger when scroller start
* @event scrollstart
*/
START: "scrollstart",
/**
* event trigger when scroller move
* @event scrollmove
*/
MOVE: "scrollmove",
/**
* event trigger when scroller end
* @event scrollend
*/
END: "scrollend",
/**
* event trigger when scroll is cancel
* @event scrollcancel
*/
CANCEL: "scrollcancel"
},
/*
* this option is related operation of scroll bar.
* the value is true, scroll bar is shown during touching screen even if content doesn't scroll.
* the value is false, scroll bar disappear when there is no movement of the scroll bar.
*/
_keepShowingScrollbarOnTouch = false,
Scroller = function () {
};
Scroller.Orientation = {
VERTICAL: "vertical",
HORIZONTAL: "horizontal"
};
Scroller.EventType = eventType;
prototype._build = function (element) {
if (element.children.length !== 1) {
throw "scroller has only one child.";
}
this.scroller = element.children[0];
this.scrollerStyle = this.scroller.style;
this.bouncingEffect = null;
this.scrollbar = null;
this.scrollerWidth = 0;
this.scrollerHeight = 0;
this.scrollerOffsetX = 0;
this.scrollerOffsetY = 0;
this.maxScrollX = 0;
this.maxScrollY = 0;
this.startScrollerOffsetX = 0;
this.startScrollerOffsetY = 0;
this.orientation = null;
this.enabled = true;
this.scrolled = false;
this.dragging = false;
this.scrollCanceled = false;
return element;
};
prototype._configure = function () {
/**
* @property {Object} options Options for widget
* @property {number} [options.scrollDelay=0]
* @property {number} [options.threshold=10]
* @property {""|"bar"|"tab"} [options.scrollbar=""]
* @property {boolean} [options.useBouncingEffect=false]
* @property {"vertical"|"horizontal"} [options.orientation="vertical"]
* @member ns.widget.core.Scroller
*/
this.options = utilsObject.merge({}, this.options, {
scrollDelay: 0,
threshold: 30,
scrollbar: "",
useBouncingEffect: false,
orientation: "vertical" // vertical or horizontal,
});
};
prototype._init = function (element) {
var options = this.options,
scrollerChildren = this.scroller.children,
elementStyle = this.element.style,
scrollerStyle = this.scroller.style,
elementHalfWidth = this.element.offsetWidth / 2,
elementHalfHeight = this.element.offsetHeight / 2;
this.orientation = this.orientation ? this.orientation :
(options.orientation === "horizontal" ? Scroller.Orientation.HORIZONTAL : Scroller.Orientation.VERTICAL);
this.scrollerWidth = this.scroller.offsetWidth;
this.scrollerHeight = this.scroller.offsetHeight;
this.maxScrollX = elementHalfWidth - this.scrollerWidth + scrollerChildren[scrollerChildren.length - 1].offsetWidth / 2;
this.maxScrollY = elementHalfHeight - this.scrollerHeight + scrollerChildren[scrollerChildren.length - 1].offsetHeight / 2;
this.minScrollX = elementHalfWidth - scrollerChildren[0].offsetWidth / 2;
this.minScrollY = elementHalfHeight - scrollerChildren[0].offsetHeight / 2;
this.scrolled = false;
this.touching = true;
this.scrollCanceled = false;
if ( this.orientation === Scroller.Orientation.HORIZONTAL ) {
this.maxScrollY = 0;
} else {
this.maxScrollX = 0;
}
elementStyle.overflow = "hidden";
elementStyle.position = "relative";
scrollerStyle.position = "absolute";
scrollerStyle.top = "0px";
scrollerStyle.left = "0px";
scrollerStyle.width = this.scrollerWidth + "px";
scrollerStyle.height = this.scrollerHeight + "px";
this._initScrollbar();
this._initBouncingEffect();
return element;
};
prototype._initScrollbar = function () {
var type = this.options.scrollbar,
scrollbarType;
if ( type ) {
scrollbarType = ns.widget.core.scroller.scrollbar.type[type];
if ( scrollbarType ) {
this.scrollbar = engine.instanceWidget(this.element, "ScrollBar", {
type: scrollbarType,
orientation: this.orientation
});
}
}
};
prototype._initBouncingEffect = function () {
var o = this.options;
if ( o.useBouncingEffect ) {
this.bouncingEffect = new EffectBouncing(this.element, {
maxScrollX: this.maxScrollX,
maxScrollY: this.maxScrollY,
orientation: this.orientation
});
}
};
prototype._resetLayout = function () {
var elementStyle = this.element.style,
scrollerStyle = this.scrollerStyle;
elementStyle.overflow = "";
elementStyle.position = "";
elementStyle.overflow = "hidden";
elementStyle.position = "relative";
if (scrollerStyle) {
scrollerStyle.position = "";
scrollerStyle.top = "";
scrollerStyle.left = "";
scrollerStyle.width = "";
scrollerStyle.height = "";
scrollerStyle["-webkit-transform"] = "";
scrollerStyle["-moz-transition"] = "";
scrollerStyle["-ms-transition"] = "";
scrollerStyle["-o-transition"] = "";
scrollerStyle["transition"] = "";
}
};
prototype._bindEvents = function () {
ns.event.enableGesture(
this.scroller,
new ns.event.gesture.Drag({
threshold: this.options.threshold,
delay: this.options.scrollDelay,
blockVertical: this.orientation === Scroller.Orientation.HORIZONTAL,
blockHorizontal: this.orientation === Scroller.Orientation.VERTICAL
})
);
utilsEvents.on( this.scroller, "drag dragstart dragend dragcancel", this );
window.addEventListener("resize", this);
};
prototype._unbindEvents = function () {
if (this.scroller) {
ns.event.disableGesture( this.scroller );
utilsEvents.off( this.scroller, "drag dragstart dragend dragcancel", this );
window.removeEventListener("resize", this);
}
};
/* jshint -W086 */
prototype.handleEvent = function (event) {
switch (event.type) {
case "dragstart":
this._start( event );
break;
case "drag":
this._move( event );
break;
case "dragend":
this._end( event );
break;
case "dragcancel":
this.cancel( event );
break;
case "resize":
this.refresh();
break;
}
};
/**
* Set options for widget.
* @method setOptions
* @param {Object} options
* @member ns.widget.core.scroller.Scroller
*/
prototype.setOptions = function (options) {
var name;
for ( name in options ) {
if ( options.hasOwnProperty(name) && !!options[name] ) {
this.options[name] = options[name];
}
}
};
prototype._refresh = function () {
this._clear();
this._unbindEvents();
this._init(this.element);
this._bindEvents();
};
/**
* Scrolls to new position.
* @method scrollTo
* @param x
* @param y
* @param duration
* @member ns.widget.core.scroller.Scroller
*/
prototype.scrollTo = function (x, y, duration) {
this._translate(x, y, duration);
this._translateScrollbar(x, y, duration);
};
prototype._translate = function (x, y, duration) {
var translate,
transition = {
normal: "none",
webkit: "none",
moz: "none",
ms: "none",
o: "none"
},
scrollerStyle = this.scrollerStyle;
if (duration) {
transition.normal = "transform " + duration / 1000 + "s ease-out";
transition.webkit = "-webkit-transform " + duration / 1000 + "s ease-out";
transition.moz = "-moz-transform " + duration / 1000 + "s ease-out";
transition.ms = "-ms-transform " + duration / 1000 + "s ease-out";
transition.o = "-o-transform " + duration / 1000 + "s ease-out";
}
translate = "translate3d(" + x + "px," + y + "px, 0)";
scrollerStyle["-webkit-transform"] =
scrollerStyle["-moz-transform"] =
scrollerStyle["-ms-transform"] =
scrollerStyle["-o-transform"] =
scrollerStyle.transform = translate;
scrollerStyle.transition = transition.normal;
scrollerStyle["-webkit-transition"] = transition.webkit;
scrollerStyle["-moz-transition"] = transition.moz;
scrollerStyle["-ms-transition"] = transition.ms;
scrollerStyle["-o-transition"] = transition.o;
this.scrollerOffsetX = window.parseInt(x, 10);
this.scrollerOffsetY = window.parseInt(y, 10);
};
prototype._translateScrollbar = function (x, y, duration, autoHidden) {
if (!this.scrollbar) {
return;
}
this.scrollbar.translate(this.orientation === Scroller.Orientation.HORIZONTAL ? -x : -y, duration, autoHidden);
};
prototype._start = function(/* e */) {
this.scrolled = false;
this.dragging = true;
this.scrollCanceled = false;
this.startScrollerOffsetX = this.scrollerOffsetX;
this.startScrollerOffsetY = this.scrollerOffsetY;
};
prototype._move = function (e, pos) {
var newX = this.startScrollerOffsetX,
newY = this.startScrollerOffsetY,
autoHide = !_keepShowingScrollbarOnTouch;
if ( !this.enabled || this.scrollCanceled || !this.dragging ) {
return;
}
if ( this.orientation === Scroller.Orientation.HORIZONTAL ) {
newX += e.detail.estimatedDeltaX;
} else {
newY += e.detail.estimatedDeltaY;
}
if ( newX > this.minScrollX || newX < this.maxScrollX ) {
newX = newX > this.minScrollX ? this.minScrollX : this.maxScrollX;
}
if ( newY > this.minScrollY || newY < this.maxScrollY ) {
newY = newY > this.minScrollY ? this.minScrollY : this.maxScrollY;
}
if ( newX !== this.scrollerOffsetX || newY !== this.scrollerOffsetY ) {
if ( !this.scrolled ) {
this._fireEvent( eventType.START );
}
this.scrolled = true;
this._translate( newX, newY );
this._translateScrollbar( newX, newY, 0, autoHide );
// TODO to dispatch move event is too expansive. it is better to use callback.
this._fireEvent( eventType.MOVE );
if ( this.bouncingEffect ) {
this.bouncingEffect.hide();
}
} else {
if ( this.bouncingEffect ) {
this.bouncingEffect.drag( newX, newY );
}
this._translateScrollbar( newX, newY, 0, autoHide );
}
};
prototype._end = function (/* e */) {
if ( !this.dragging ) {
return;
}
// bouncing effect
if ( this.bouncingEffect ) {
this.bouncingEffect.dragEnd();
}
if ( this.scrollbar ) {
this.scrollbar.end();
}
this._endScroll();
this.dragging = false;
};
prototype._endScroll = function () {
if (this.scrolled) {
this._fireEvent(eventType.END);
}
this.scrolled = false;
};
/**
* Cancels scroll.
* @method cancel
* @member ns.widget.core.scroller.Scroller
*/
prototype.cancel = function () {
this.scrollCanceled = true;
if ( this.scrolled ) {
this._translate( this.startScrollerOffsetX, this.startScrollerOffsetY );
this._translateScrollbar( this.startScrollerOffsetX, this.startScrollerOffsetY );
this._fireEvent( eventType.CANCEL );
}
if ( this.scrollbar ) {
this.scrollbar.end();
}
this.scrolled = false;
this.dragging = false;
};
prototype._fireEvent = function (eventName, detail) {
eventTrigger( this.element, eventName, detail );
};
prototype._clear = function () {
this.scrolled = false;
this.scrollCanceled = false;
this._resetLayout();
this._clearScrollbar();
this._clearBouncingEffect();
};
prototype._clearScrollbar = function () {
if ( this.scrollbar ) {
this.scrollbar.destroy();
}
this.scrollbar = null;
};
prototype._clearBouncingEffect = function () {
if (this.bouncingEffect) {
this.bouncingEffect.destroy();
}
this.bouncingEffect = null;
};
prototype._disable = function () {
this.enabled = false;
};
prototype._enable = function () {
this.enabled = true;
};
prototype._destroy = function () {
this._clear();
this._unbindEvents();
this.scrollerStyle = null;
this.scroller = null;
};
Scroller.prototype = prototype;
ns.widget.core.scroller.Scroller = Scroller;
engine.defineWidget(
"Scroller",
".scroller",
["scrollTo", "cancel"],
Scroller
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* #Scrollbar namespace
* Namespace with scrollbar for scroller widget.
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @class ns.widget.core.scroller.scrollbar
*/
(function (window, ns) {
"use strict";
ns.widget.core.scroller.scrollbar = ns.widget.core.scroller.scrollbar || {};
}(window, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* #type namespace
* Namespace with types of scroll bars..
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @class ns.widget.core.scroller.scrollbar.type
*/
(function (window, ns) {
"use strict";
/** @namespace ns.widget.core */
ns.widget.core.scroller.scrollbar.type = ns.widget.core.scroller.scrollbar.type || {};
}(window, ns));
/*global window, define, Event, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #Type Interface
* Interface for types used in scroll bar widget.
* @class ns.widget.core.scroller.scrollbar.type.interface
*/
(function (document, ns) {
"use strict";
// scroller.start event trigger when user try to move scroller
ns.widget.core.scroller.scrollbar.type.interface = {
/**
* Inserts elements end decorate.
* @method insertAndDecorate
* @param options
* @static
* @member ns.widget.core.scroller.scrollbar.type.interface
*/
setScrollbarLayout: function (/* options */) {
},
/**
* Removes element.
* @method remove
* @param options
* @static
* @member ns.widget.core.scroller.scrollbar.type.interface
*/
remove: function (/* options */) {
},
/**
* ...
* @method start
* @param scrollbarElement
* @param barElement
* @static
* @member ns.widget.core.scroller.scrollbar.type.interface
*/
start: function (/* scrollbarElement, barElement */) {
},
/**
* ...
* @method end
* @param scrollbarElement
* @param barElement
* @static
* @member ns.widget.core.scroller.scrollbar.type.interface
*/
end: function (/* scrollbarElement, barElement */) {
},
/**
* ...
* @method offset
* @param orientation
* @param offset
* @static
* @member ns.widget.core.scroller.scrollbar.type.interface
*/
offset: function (/* orientation, offset */) {
}
};
}(window.document, ns));
/*global window, define, Event, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #Bar Type
* Bar type support for scroll bar widget.
* @class ns.widget.core.scroller.scrollbar.type.bar
* @extends ns.widget.core.scroller.scrollbar.type.interface
*/
(function (document, ns) {
"use strict";
// scroller.start event trigger when user try to move scroller
var utilsObject = ns.util.object,
type = ns.widget.core.scroller.scrollbar.type,
typeInterface = type.interface,
Scroller = ns.widget.core.scroller.Scroller;
type.bar = utilsObject.merge({}, typeInterface, {
options: {
animationDuration: 500
},
/**
* @method setScrollbar
* @param viewLayout
* @param firstChildLayout
* @param clipLayout
* @static
* @member ns.widget.core.scroller.scrollbar.type.bar
*/
setScrollbar: function(viewLayout, firstChildLayout, clipLayout) {
this._viewLayout = viewLayout;
this._clipLayout = clipLayout;
this._firstChildLayout = firstChildLayout;
this._ratio = clipLayout / firstChildLayout;
},
/**
* @method getScrollbarSize
* @return scrollbar size
* @static
* @member ns.widget.core.scroller.scrollbar.type.bar
*/
getScrollbarSize: function() {
return this._firstChildLayout / this._viewLayout * this._firstChildLayout * this._ratio;
},
/**
* @method offset
* @param orientation
* @param offset
* @static
* @member ns.widget.core.scroller.scrollbar.type.bar
*/
offset: function( orientation, offset ) {
var x, y;
offset = offset * this._clipLayout / this._viewLayout;
if ( orientation === Scroller.Orientation.VERTICAL ) {
x = 0;
y = offset;
} else {
x = offset;
y = 0;
}
return {
x: x,
y: y
};
},
/**
* @method start
* @param scrollbarElement
* @static
* @member ns.widget.core.scroller.scrollbar.type.bar
*/
start: function( scrollbarElement/*, barElement */) {
var style = scrollbarElement.style,
duration = this.options.animationDuration;
style["-webkit-transition"] =
style["-moz-transition"] =
style["-ms-transition"] =
style["-o-transition"] =
style.transition = "opacity " + duration / 1000 + "s ease";
style.opacity = 1;
},
/**
* @method end
* @param scrollbarElement
* @static
* @member ns.widget.core.scroller.scrollbar.type.bar
*/
end: function( scrollbarElement/*, barElement */) {
var style = scrollbarElement.style,
duration = this.options.animationDuration;
style["-webkit-transition"] =
style["-moz-transition"] =
style["-ms-transition"] =
style["-o-transition"] =
style.transition = "opacity " + duration / 1000 + "s ease";
style.opacity = 0;
}
});
}(window.document, ns));
/*global window, define, Event, console, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #Scroll Bar Widget
* Widget creates scroll bar.
* @class ns.widget.core.scroller.scrollbar.ScrollBar
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
// scroller.start event trigger when user try to move scroller
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
prototype = new BaseWidget(),
utilsObject = ns.util.object,
selectors = ns.util.selectors,
Page = ns.widget.core.Page,
Classes = {
wrapperClass: "ui-scrollbar-bar-type",
barClass: "ui-scrollbar-indicator",
orientationClass: "ui-scrollbar-",
page: Page.classes.uiPage
},
Scroller = ns.widget.core.scroller.Scroller,
ScrollerScrollBar = function () {
this.wrapper = null;
this.barElement = null;
this.container = null;
this.view = null;
this.options = {};
this.type = null;
this.maxScroll = null;
this.started = false;
this.displayDelayTimeoutId = null;
this.lastScrollPosition = 0;
};
prototype._build = function (scrollElement) {
this.clip = scrollElement;
this.view = scrollElement.children[0];
this.firstChild = this.view.children[0];
return scrollElement;
};
prototype._configure = function () {
/**
* @property {Object} options Options for widget
* @property {boolean} [options.type=false]
* @property {number} [options.displayDelay=700]
* @property {"vertical"|"horizontal"} [options.orientation="vertical"]
* @member ns.widget.core.scroller.scrollbar.ScrollBar
*/
this.options = utilsObject.merge({}, this.options, {
type: false,
displayDelay: 700,
orientation: Scroller.Orientation.VERTICAL
});
};
prototype._init = function () {
this.type = this.options.type;
if ( !this.type ) {
return;
}
this._createScrollbar();
};
prototype._bindEvents = function() {
document.addEventListener("visibilitychange", this);
};
prototype._createScrollbar = function () {
var orientation = this.options.orientation,
wrapper = document.createElement("DIV"),
bar = document.createElement("span"),
view = this.view,
clip = this.clip,
firstChild = this.firstChild,
type = this.type;
clip.appendChild(wrapper);
wrapper.appendChild(bar);
wrapper.classList.add(Classes.wrapperClass);
bar.className = Classes.barClass;
if (orientation === Scroller.Orientation.HORIZONTAL) {
type.setScrollbar(view.offsetWidth, firstChild.offsetWidth, clip.offsetWidth);
bar.style.width = type.getScrollbarSize() + "px";
wrapper.classList.add(Classes.orientationClass + "horizontal");
} else {
type.setScrollbar(view.offsetHeight, firstChild.offsetHeight, clip.offsetHeight);
bar.style.height = type.getScrollbarSize() + "px";
wrapper.classList.add(Classes.orientationClass + "vertical");
}
this.wrapper = wrapper;
this.barElement = bar;
};
prototype._removeScrollbar = function () {
this.clip.removeChild(this.wrapper);
this.wrapper = null;
this.barElement = null;
};
prototype._refresh = function () {
var self = this;
self._clear();
self._init();
self.translate(self.lastScrollPosition);
};
/**
* Translates widget.
* @method translate
* @param offset
* @param duration
* @member ns.widget.core.scroller.scrollbar.ScrollBar
*/
prototype.translate = function (offset, duration, autoHidden) {
var orientation = this.options.orientation,
translate,
transition = {
normal: "none",
webkit: "none",
moz: "none",
ms: "none",
o: "none"
},
barStyle,
endDelay;
if ( !this.wrapper || !this.type || this.lastScrollPosition === offset ) {
return;
}
autoHidden = autoHidden !== false;
this.lastScrollPosition = offset;
offset = this.type.offset( orientation, offset );
barStyle = this.barElement.style;
if (duration) {
transition.normal = "transform " + duration / 1000 + "s ease-out";
transition.webkit = "-webkit-transform " + duration / 1000 + "s ease-out";
transition.moz = "-moz-transform " + duration / 1000 + "s ease-out";
transition.ms = "-ms-transform " + duration / 1000 + "s ease-out";
transition.o = "-o-transform " + duration / 1000 + "s ease-out";
}
translate = "translate3d(" + offset.x + "px," + offset.y + "px, 0)";
barStyle["-webkit-transform"] =
barStyle["-moz-transform"] =
barStyle["-ms-transform"] =
barStyle["-o-transform"] =
barStyle.transform = translate;
barStyle["-webkit-transition"] = transition.webkit;
barStyle["-moz-transition"] = transition.moz;
barStyle["-ms-transition"] = transition.ms;
barStyle["-o-transition"] = transition.o;
barStyle.transition = transition.normal;
if ( !this.started ) {
this._start();
}
if ( this.displayDelayTimeoutId !== null ) {
window.clearTimeout( this.displayDelayTimeoutId );
this.displayDelayTimeoutId = null;
}
if ( autoHidden ) {
endDelay = ( duration || 0 ) + this.options.displayDelay;
this.displayDelayTimeoutId = window.setTimeout(this._end.bind(this), endDelay);
}
};
prototype.end = function () {
if ( !this.displayDelayTimeoutId ) {
this.displayDelayTimeoutId = window.setTimeout(this._end.bind(this), this.options.displayDelay);
}
};
prototype._start = function () {
this.type.start(this.wrapper, this.barElement);
this.started = true;
};
prototype._end = function () {
this.started = false;
this.displayDelayTimeoutId = null;
if ( this.type ) {
this.type.end(this.wrapper, this.barElement);
}
};
/**
* Supports events.
* @method handleEvent
* @param event
* @member ns.widget.core.scroller.scrollbar.ScrollBar
*/
prototype.handleEvent = function(event) {
var page;
switch(event.type) {
case "visibilitychange":
page = selectors.getClosestBySelector(this.clip, "." + Classes.page);
if (document.visibilityState === "visible" && page === ns.activePage) {
this.refresh();
}
break;
}
};
prototype._clear = function () {
this._removeScrollbar();
this.started = false;
this.type = null;
this.barElement = null;
this.displayDelayTimeoutId = null;
};
prototype._destroy = function () {
this._clear();
document.removeEventListener("visibilitychange", this);
this.options = null;
this.clip = null;
this.view = null;
};
ScrollerScrollBar.prototype = prototype;
ns.widget.core.scroller.scrollbar.ScrollBar = ScrollerScrollBar;
engine.defineWidget(
"ScrollBar",
"",
["translate"],
ScrollerScrollBar
);
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Grid Utility
* Object helps creating grids.
* @class ns.util.grid
*/
(function (ns) {
"use strict";
/**
* Local alias for ns.util.selectors
* @property {Object} selectors Alias for {@link ns.util.selectors}
* @member ns.util.grid
* @static
* @private
*/
var selectors = ns.util.selectors,
/**
* Alias to Array.slice method
* @method slice
* @member ns.util.grid
* @private
* @static
*/
slice = [].slice,
/**
* grid types
* @property {Array} gridTypes
* @member ns.util.grid
* @static
* @private
*/
gridTypes = [
null,
"solo", //1
"a", //2
"b", //3
"c", //4
"d" //5
];
/**
* Add classes on the matched elements
* @method setClassOnMatches
* @param {HTMLElementCollection} elements
* @param {string} selector
* @param {string} className
* @private
* @member ns.util.grid
* @static
*/
function setClassOnMatches(elements, selector, className) {
elements.forEach(function (item) {
if (selectors.matchesSelector(item, selector)) {
item.classList.add(className);
}
});
}
ns.util.grid = {
/**
* make css grid
* @method makeGrid
* @param {HTMLElement} element
* @param {?string} [gridType="a"]
* @static
* @member ns.util.grid
*/
makeGrid: function (element, gridType) {
var gridClassList = element.classList,
kids = slice.call(element.children),
iterator;
if (!gridType) {
gridType = gridTypes[kids.length];
if (!gridType) {
//if gridType is not defined in gritTypes
//make it grid type "a""
gridType = "a";
iterator = 2;
gridClassList.add("ui-grid-duo");
}
}
if (!iterator) {
//jquery grid doesn't care if someone gives non-existing gridType
iterator = gridTypes.indexOf(gridType);
}
gridClassList.add("ui-grid-" + gridType);
setClassOnMatches(kids, ":nth-child(" + iterator + "n+1)", "ui-block-a");
if (iterator > 1) {
setClassOnMatches(kids, ":nth-child(" + iterator + "n+2)", "ui-block-b");
}
if (iterator > 2) {
setClassOnMatches(kids, ":nth-child(" + iterator + "n+3)", "ui-block-c");
}
if (iterator > 3) {
setClassOnMatches(kids, ":nth-child(" + iterator + "n+4)", "ui-block-d");
}
if (iterator > 4) {
setClassOnMatches(kids, ":nth-child(" + iterator + "n+5)", "ui-block-e");
}
}
};
}(ns));
/*global CustomEvent, define, window, ns */
/*jslint plusplus: true, nomen: true, bitwise: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Virtual Mouse Events
* Reimplementation of jQuery Mobile virtual mouse events.
*
* ##Purpose
* It will let for users to register callbacks to the standard events like bellow,
* without knowing if device support touch or mouse events
* @class ns.event.vmouse
*/
/**
* Triggered after mouse-down or touch-started.
* @event vmousedown
* @member ns.event.vmouse
*/
/**
* Triggered when mouse-click or touch-end when touch-move didn't occur
* @event vclick
* @member ns.event.vmouse
*/
/**
* Triggered when mouse-up or touch-end
* @event vmouseup
* @member ns.event.vmouse
*/
/**
* Triggered when mouse-move or touch-move
* @event vmousemove
* @member ns.event.vmouse
*/
/**
* Triggered when mouse-over or touch-start if went over coordinates
* @event vmouseover
* @member ns.event.vmouse
*/
/**
* Triggered when mouse-out or touch-end
* @event vmouseout
* @member ns.event.vmouse
*/
/**
* Triggered when mouse-cancel or touch-cancel and when scroll occur during touchmove
* @event vmousecancel
* @member ns.event.vmouse
*/
(function (window, document, ns) {
"use strict";
/**
* Object with default options
* @property {Object} vmouse
* @member ns.event.vmouse
* @static
* @private
**/
var vmouse,
/**
* @property {Object} eventProps Contains the properties which are copied from the original event to custom v-events
* @member ns.event.vmouse
* @static
* @private
**/
eventProps,
/**
* Indicates if the browser support touch events
* @property {boolean} touchSupport
* @member ns.event.vmouse
* @static
**/
touchSupport = window.hasOwnProperty("ontouchstart"),
/**
* @property {boolean} didScroll The flag tell us if the scroll event was triggered
* @member ns.event.vmouse
* @static
* @private
**/
didScroll,
/** @property {HTMLElement} lastOver holds reference to last element that touch was over
* @member ns.event.vmouse
* @private
*/
lastOver = null,
/**
* @property {Number} [startX=0] Initial data for touchstart event
* @member ns.event.vmouse
* @static
* @private
**/
startX = 0,
/**
* @property {Number} [startY=0] Initial data for touchstart event
* @member ns.event.vmouse
* @private
* @static
**/
startY = 0,
touchEventProps = ["clientX", "clientY", "pageX", "pageY", "screenX", "screenY"],
KEY_CODES = {
enter: 13
};
/**
* Extends objects with other objects
* @method copyProps
* @param {Object} from Sets the original event
* @param {Object} to Sets the new event
* @param {Object} properties Sets the special properties for position
* @param {Object} propertiesNames Describe parameters which will be copied from Original to To event
* @private
* @static
* @member ns.event.vmouse
*/
function copyProps(from, to, properties, propertiesNames) {
var i,
length,
descriptor,
property;
for (i = 0, length = propertiesNames.length; i < length; ++i) {
property = propertiesNames[i];
if (isNaN(properties[property]) === false || isNaN(from[property]) === false) {
descriptor = Object.getOwnPropertyDescriptor(to, property);
if (property !== "detail" && (!descriptor || descriptor.writable)) {
to[property] = properties[property] || from[property];
}
}
}
}
/**
* Create custom event
* @method createEvent
* @param {string} newType gives a name for the new Type of event
* @param {Event} original Event which trigger the new event
* @param {Object} properties Sets the special properties for position
* @return {Event}
* @private
* @static
* @member ns.event.vmouse
*/
function createEvent(newType, original, properties) {
var evt = new CustomEvent(newType, {
"bubbles": original.bubbles,
"cancelable": original.cancelable,
"detail": original.detail
}),
orginalType = original.type,
changeTouches,
touch,
j = 0,
len,
prop;
copyProps(original, evt, properties, eventProps);
evt._originalEvent = original;
if (orginalType.indexOf("touch") !== -1) {
orginalType = original.touches;
changeTouches = original.changedTouches;
if (orginalType && orginalType.length) {
touch = orginalType[0];
} else {
touch = (changeTouches && changeTouches.length) ? changeTouches[0] : null;
}
if (touch) {
for (len = touchEventProps.length; j < len; j++) {
prop = touchEventProps[j];
evt[prop] = touch[prop];
}
}
}
return evt;
}
/**
* Dispatch Events
* @method fireEvent
* @param {string} eventName event name
* @param {Event} evt original event
* @param {Object} [properties] Sets the special properties for position
* @return {boolean}
* @private
* @static
* @member ns.event.vmouse
*/
function fireEvent(eventName, evt, properties) {
return evt.target.dispatchEvent(createEvent(eventName, evt, properties || {}));
}
eventProps = [
"currentTarget",
"detail",
"button",
"buttons",
"clientX",
"clientY",
"offsetX",
"offsetY",
"pageX",
"pageY",
"screenX",
"screenY",
"toElement",
"which"
];
vmouse = {
/**
* Sets the distance of pixels after which the scroll event will be successful
* @property {number} [eventDistanceThreshold=10]
* @member ns.event.vmouse
* @static
*/
eventDistanceThreshold: 10,
touchSupport: touchSupport
};
/**
* Handle click down
* @method handleDown
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleDown(evt) {
fireEvent("vmousedown", evt);
}
/**
* Prepare position of event for keyboard events.
* @method preparePositionForClick
* @param {Event} event
* @return {?Object} options
* @private
* @static
* @member ns.event.vmouse
*/
function preparePositionForClick(event) {
var x = event.clientX,
y = event.clientY;
// event comes from keyboard
if (!x && !y) {
return preparePositionForEvent(event);
}
}
/**
* Handle click
* @method handleClick
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleClick(evt) {
fireEvent("vclick", evt, preparePositionForClick(evt));
}
/**
* Handle click up
* @method handleUp
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleUp(evt) {
fireEvent("vmouseup", evt);
}
/**
* Handle click move
* @method handleMove
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleMove(evt) {
fireEvent("vmousemove", evt);
}
/**
* Handle click over
* @method handleOver
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleOver(evt) {
fireEvent("vmouseover", evt);
}
/**
* Handle click out
* @method handleOut
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleOut(evt) {
fireEvent("vmouseout", evt);
}
/**
* Handle touch start
* @method handleTouchStart
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleTouchStart(evt) {
var touches = evt.touches,
firstTouch,
over;
//if touches are registered and we have only one touch
if (touches && touches.length === 1) {
didScroll = false;
firstTouch = touches[0];
startX = firstTouch.pageX;
startY = firstTouch.pageY;
// Check if we have touched something on our page
// @TODO refactor for multi touch
over = document.elementFromPoint(startX, startY);
if (over) {
lastOver = over;
fireEvent("vmouseover", evt);
}
fireEvent("vmousedown", evt);
}
}
/**
* Handle touch end
* @method handleTouchEnd
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleTouchEnd(evt) {
var touches = evt.touches;
if (touches && touches.length === 0) {
fireEvent("vmouseup", evt);
fireEvent("vmouseout", evt);
// Reset flag for last over element
lastOver = null;
}
}
/**
* Handle touch move
* @method handleTouchMove
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleTouchMove(evt) {
var over,
firstTouch = evt.touches && evt.touches[0],
didCancel = didScroll,
//sets the threshold, based on which we consider if it was the touch-move event
moveThreshold = vmouse.eventDistanceThreshold;
/**
* Ignore the touch which has identifier other than 0.
* Only first touch has control others are ignored.
* Patch for webkit behaviour where touchmove event
* is triggered between touchend events
* if there is multi touch.
*/
if (firstTouch.identifier > 0) {
evt.preventDefault();
evt.stopPropagation();
return;
}
didScroll = didScroll ||
//check in both axes X,Y if the touch-move event occur
(Math.abs(firstTouch.pageX - startX) > moveThreshold ||
Math.abs(firstTouch.pageY - startY) > moveThreshold);
// detect over event
// for compatibility with mouseover because "touchenter" fires only once
// @TODO Handle many touches
over = document.elementFromPoint(firstTouch.pageX, firstTouch.pageY);
if (over && lastOver !== over) {
lastOver = over;
fireEvent("vmouseover", evt);
}
//if didscroll occur and wasn't canceled then trigger touchend otherwise just touchmove
if (didScroll && !didCancel) {
fireEvent("vmousecancel", evt);
lastOver = null;
}
fireEvent("vmousemove", evt);
}
/**
* Handle Scroll
* @method handleScroll
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleScroll(evt) {
if (!didScroll) {
fireEvent("vmousecancel", evt);
}
didScroll = true;
}
/**
* Handle touch cancel
* @method handleTouchCancel
* @param {Event} evt
* @private
* @static
* @member ns.event.vmouse
*/
function handleTouchCancel(evt) {
fireEvent("vmousecancel", evt);
lastOver = null;
}
/**
* Prepare position of event for keyboard events.
* @method preparePositionForEvent
* @param {Event} event
* @return {Object} properties
* @private
* @static
* @member ns.event.vmouse
*/
function preparePositionForEvent(event) {
var targetRect = event.target && event.target.getBoundingClientRect(),
properties = {};
if (targetRect) {
properties = {
"clientX": targetRect.left + targetRect.width / 2,
"clientY": targetRect.top + targetRect.height / 2,
"which": 1
};
}
return properties;
}
/**
* Handle key up
* @method handleKeyUp
* @param {Event} event
* @private
* @static
* @member ns.event.vmouse
*/
function handleKeyUp(event) {
var properties;
if (event.keyCode === KEY_CODES.enter) {
properties = preparePositionForEvent(event);
fireEvent("vmouseup", event, properties);
fireEvent("vclick", event, properties);
}
}
/**
* Handle key down
* @method handleKeyDown
* @param {Event} event
* @private
* @static
* @member ns.event.vmouse
*/
function handleKeyDown(event) {
if (event.keyCode === KEY_CODES.enter) {
fireEvent("vmousedown", event, preparePositionForEvent(event));
}
}
/**
* Binds events common to mouse and touch to support virtual mouse.
* @method bindCommonEvents
* @static
* @member ns.event.vmouse
*/
vmouse.bindCommonEvents = function () {
document.addEventListener("keyup", handleKeyUp, true);
document.addEventListener("keydown", handleKeyDown, true);
document.addEventListener("scroll", handleScroll, true);
document.addEventListener("click", handleClick, true);
};
// @TODO delete touchSupport flag and attach touch and mouse listeners,
// @TODO check if v-events are not duplicated if so then called only once
/**
* Binds touch events to support virtual mouse.
* @method bindTouch
* @static
* @member ns.event.vmouse
*/
vmouse.bindTouch = function () {
document.addEventListener("touchstart", handleTouchStart, true);
document.addEventListener("touchend", handleTouchEnd, true);
document.addEventListener("touchmove", handleTouchMove, true);
document.addEventListener("touchcancel", handleTouchCancel, true);
// touchenter and touchleave are removed from W3C spec
// No need to listen to touchover as it has never exited
// document.addEventListener("touchenter", handleTouchOver, true);
// document.addEventListener("touchleave", callbacks.out, true);
document.addEventListener("touchcancel", handleTouchCancel, true);
};
/**
* Binds mouse events to support virtual mouse.
* @method bindMouse
* @static
* @member ns.event.vmouse
*/
vmouse.bindMouse = function () {
document.addEventListener("mousedown", handleDown, true);
document.addEventListener("mouseup", handleUp, true);
document.addEventListener("mousemove", handleMove, true);
document.addEventListener("mouseover", handleOver, true);
document.addEventListener("mouseout", handleOut, true);
};
ns.event.vmouse = vmouse;
if (touchSupport) {
vmouse.bindTouch();
} else {
vmouse.bindMouse();
}
vmouse.bindCommonEvents();
}(window, window.document, ns));
/*global window, define */
/*jslint nomen: true, plusplus: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Easing Utility
* Utility calculates time function for animations.
* @class ns.util.easing
*/
(function (ns) {
"use strict";
ns.util.easing = {
/**
* Performs cubit out easing calcuclations based on time
* @method cubicOut
* @member ns.util.easing
* @param {number} currentTime
* @param {number} startValue
* @param {number} changeInValue
* @param {number} duration
* @return {number}
* @static
*/
cubicOut: function (currentTime, startValue, changeInValue, duration) {
currentTime /= duration;
currentTime--;
return changeInValue * (currentTime * currentTime * currentTime + 1) + startValue;
},
/**
* Performs quad easing out calcuclations based on time
* @method easeOutQuad
* @member ns.util.easing
* @param {number} currentTime
* @param {number} startValue
* @param {number} changeInValue
* @param {number} duration
* @return {number}
* @static
*/
easeOutQuad: function (currentTime, startValue, changeInValue, duration) {
return -changeInValue * (currentTime /= duration) * (currentTime - 2) + startValue;
},
/**
* Performs out expo easing calcuclations based on time
* @method easeOutExpo
* @member ns.util.easing
* @param {number} currentTime
* @param {number} startValue
* @param {number} changeInValue
* @param {number} duration
* @return {number}
* @static
*/
easeOutExpo: function (currentTime, startValue, changeInValue, duration) {
return (currentTime === duration) ?
startValue + changeInValue :
changeInValue * (-Math.pow(2, -10 * currentTime / duration) + 1) +
startValue;
},
/**
* Performs out linear calcuclations based on time
* @method linear
* @member ns.util.easing
* @param {number} currentTime
* @param {number} startValue
* @param {number} changeInValue
* @param {number} duration
* @return {number}
* @static
*/
linear: function (currentTime, startValue, changeInValue, duration) {
return startValue + duration * currentTime;
}
};
}(ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Event orientationchange
* Namespace to support orientationchange event
* @class ns.event.orientationchange
*/
/**
* Event orientationchange
* @event orientationchange
* @member ns.event.orientationchange
*/
(function (window, document, ns) {
"use strict";
var body = document.body,
orientation = null,
eventUtils = ns.event,
orientationchange = {
/**
* Informs about support orientation change event.
* @property {boolean} supported
* @member ns.event.orientationchange
*/
supported: (window.orientation !== undefined) && (window.onorientationchange !== undefined),
/**
* Returns current orientation.
* @method getOrientation
* @return {"landscape"|"portrait"}
* @member ns.event.orientationchange
* @static
*/
getOrientation: function () {
return orientation;
},
/**
* Triggers event orientationchange on element
* @method trigger
* @param {HTMLElement} element
* @member ns.event.orientationchange
* @static
*/
trigger: function (element) {
eventUtils.trigger(element, "orientationchange", {'orientation': orientation});
},
/**
* List of properties copied to event details object
* @property {Array} properties
* @member ns.event.orientationchange
* @static
*/
properties: ['orientation']
},
detectOrientationByDimensions = function (omitCustomEvent) {
var width = window.innerWidth,
height = window.innerHeight;
if (window.screen) {
width = window.screen.availWidth;
height = window.screen.availHeight;
}
if (width > height) {
orientation = "landscape";
} else {
orientation = "portrait";
}
if (!omitCustomEvent) {
eventUtils.trigger(window, "orientationchange", {'orientation': orientation});
}
},
checkReportedOrientation = function () {
if (window.orientation) {
switch (window.orientation) {
case 90:
case -90:
orientation = "portrait";
break;
default:
orientation = "landscape";
break;
}
} else {
detectOrientationByDimensions(true);
}
},
matchMediaHandler = function (mediaQueryList) {
if (mediaQueryList.matches) {
orientation = "portrait";
} else {
orientation = "landscape";
}
eventUtils.trigger(window, "orientationchange", {'orientation': orientation});
},
portraitMatchMediaQueryList;
if (orientationchange.supported) {
window.addEventListener("orientationchange", checkReportedOrientation, false);
checkReportedOrientation();
// try media queries
} else {
if (window.matchMedia) {
portraitMatchMediaQueryList = window.matchMedia("(orientation: portrait)");
if (portraitMatchMediaQueryList.matches) {
orientation = "portrait";
} else {
orientation = "landscape";
}
portraitMatchMediaQueryList.addListener(matchMediaHandler);
} else {
body.addEventListener("throttledresize", detectOrientationByDimensions);
detectOrientationByDimensions();
}
}
ns.event.orientationchange = orientationchange;
}(window, window.document, ns));
/*global window, define, ns*/
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* # ScrollView Widget
* Widgets allows for creating scrollable panes, lists, etc.
*
* ## Default selectors
* All elements with _data-role=content attribute or _.ui-scrollview
* css class will be changed to ScrollView widgets, unless they specify
* _data-scroll=none attribute.
*
* ### HTML Examples
*
* #### Data attribute
*
* @example
* <div data-role="page">
* <div data-role="content"><!-- this will become scrollview //-->
* content data
* </div>
* </div>
*
* #### CSS Class
*
* @example
* <div data-role="page">
* <div class="ui-content"><!-- this will become scrollview //-->
* content data
* </div>
* </div>
*
* ## Manual constructor
*
* To create the widget manually you can use 2 different APIs, the TAU
* API or jQuery API.
*
* ### Create scrollview by TAU API
*
* @example
* <div data-role="page" id="myPage">
* <div data-role="content">
* page content
* </div>
* </div>
* <script>
* var page = tau.widget.Page(document.getElementById("myPage")),
* scrollview = tau.widget.Scrollview(page.ui.content);
* </script>
*
* ### Create scrollview using jQuery API
*
* @example
* <div data-role="page" id="myPage">
* <div data-role="content">
* page content
* </div>
* </div>
* <script>
* $("#myPage > div[data-role='content']").scrollview();
* </script>
*
* ## Options for Scrollview widget
*
* Options can be set using data-* attributes or by passing them to
* the constructor.
*
* There is also a method **option** for changing them after widget
* creation.
*
* jQuery mobile format is also supported.
*
* ## Scroll
*
* This options specifies of a content element should become Scrollview
* widget.
*
* You can change this by all available methods for changing options.
*
* ### By data-scroll attribute
*
* @example
* <div data-role="page">
* <div data-role="content" data-scroll="none">
* content
* </div>
* </div>
*
* ### By config passed to constructor
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content">
* content
* </div>
* </div>
* <script>
* var contentElement = document.querySelector(".myPageClass > div[data-role=content]");
* tau.widget.Scrollview(contentElement, {
* "scroll": false
* });
* </script>
*
* ### By using jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content">
* content
* </div>
* </div>
* <script>
* $(".myPageClass > div[data-role='content']").scrollview({
* "scroll": false
* });
* </script>
*
* ## ScrollJumps
*
* Scroll jumps are small buttons which allow the user to quickly
* scroll to top or left
*
* You can change this by all available methods for changing options.
*
* ### By data-scroll-jump
*
* @example
* <div data-role="page">
* <div data-role="content" data-scroll-jump="true">
* content
* </div>
* </div>
*
* ### By config passed to constructor
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content">
* content
* </div>
* </div>
* <script>
* var contentElement = document.querySelector(".myPageClass > div[data-role=content]");
* tau.widget.Scrollview(contentElement, {
* "scrollJump": true
* });
* </script>
*
* ### By using jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content">
* content
* </div>
* </div>
* <script>
* $(".myPageClass > div[data-role='content']").scrollview({
* "scrollJump": true
* });
* </script>
*
* ## Methods
*
* Page methods can be called trough 2 APIs: TAU API and jQuery API
* (jQuery mobile-like API)
*
* @class ns.widget.core.Scrollview
* @extends ns.widget.BaseWidget
*
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Grzegorz Osimowicz <g.osimowicz@samsung.com>
* @author Jadwiga Sosnowska <j.sosnowska@samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Hyunkook Cho <hk0713.cho@samsung.com>
* @author Junhyeon Lee <juneh.lee@samsung.com>
*/
/**
* Triggered when scrolling operation starts
* @event scrollstart
* @member ns.widget.core.Scrollview
*/
/**
* Triggered when scroll is being updated
* @event scrollupdate
* @member ns.widget.core.Scrollview
*/
/**
* Triggered when scrolling stops
* @event scrollstop
* @member ns.widget.core.Scrollview
*/
(function (window, document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
util = ns.util,
easingUtils = ns.util.easing,
eventUtils = ns.event,
DOMUtils = ns.util.DOM,
selectors = ns.util.selectors,
currentTransition = null,
Page = ns.widget.core.Page,
pageClass = Page.classes.uiPage,
pageActiveClass = Page.classes.uiPageActive,
Scrollview = function () {
var self = this,
ui;
/**
* @property {Object} state Scrollview internal state object
* @property {Function} state.currentTransition Instance transition function
* @readonly
*/
self.state = {
currentTransition: null
};
/**
* @property {number} scrollDuration The time length of the scroll animation
* @member ns.widget.core.Scrollview
*/
self.scrollDuration = 300;
self.scrollviewSetHeight = false;
/**
* Scrollview options
* @property {Object} options
* @property {string} [options.scroll='y'] Scroll direction
* @property {boolean} [options.scrollJump=false] Scroll jump buttons flag
* @member ns.widget.core.Scrollview
*/
self.options = {
scroll: "y",
scrollJump: false,
scrollIndicator: false
};
/**
* Dictionary for holding internal DOM elements
* @property {Object} ui
* @property {HTMLElement} ui.view The main view element
* @property {HTMLElement} ui.page The main page element
* @property {HTMLElement} ui.jumpHorizontalButton Jump left button
* @property {HTMLElement} ui.jumpVerticalButton Jump top button
* @member ns.widget.core.Scrollview
* @readonly
*/
ui = self._ui || {};
ui.view = null;
ui.page = null;
ui.jumpHorizontalButton = null;
ui.jumpVerticalButton = null;
self._ui = ui;
/**
* Dictionary for holding internal listeners
* @property {Object} _callbacks
* @property {Function} _callbacks.repositionJumps Refresh jumps listener
* @property {Function} _callbacks.jumpTop Top jump button click callback
* @property {Function} _callbacks.jumpLeft Left jump button click callback
* @member ns.widget.core.Scrollview
* @protected
* @readonly
*/
self._callbacks = {
repositionJumps: null,
jumpTop: null,
jumpBottom: null
};
self._timers = {
scrollIndicatorHide: null
};
},
/**
* Dictionary for scrollview css classes
* @property {Object} classes
* @property {string} [classes.view='ui-scrollview-view'] View main class
* @property {string} [classes.clip='ui-scrollview-clip'] Clip main class
* @property {string} [classes.jumpTop='ui-scroll-jump-top-bg'] Jump top button background
* @property {string} [classes.jumpLeft='ui-scroll-jump-left-bg'] Jump bottom button background
* @member ns.widget.core.Scrollview
* @static
* @readonly
*/
classes = {
view: "ui-scrollview-view",
clip: "ui-scrollview-clip",
jumpTop: "ui-scroll-jump-top-bg",
jumpLeft: "ui-scroll-jump-left-bg",
indicatorTop: "ui-overflow-indicator-top",
indicatorBottom: "ui-overflow-indicator-bottom",
indicatorTopShown: "ui-scrollindicator-top",
indicatorBottomShown: "ui-scrollindicator-bottom",
indicatorLeftShown: "ui-scrollindicator-left",
indicatorRightShown: "ui-scrollindicator-right"
};
// Changes static position to relative
// @param {HTMLElement} view
function makePositioned(view) {
if (DOMUtils.getCSSProperty(view, "position") === "static") {
view.style.position = "relative";
} else {
view.style.position = "absolute";
}
}
// Translation animation loop
// @param {Object} state Scrollview instance state
// @param {HTMLElement} element
// @param {number} startTime
// @param {number} startX
// @param {number} startY
// @param {number} translateX
// @param {number} translateY
// @param {number} endX
// @param {number} endY
// @param {number} duration
function translateTransition(state, element, startTime, startX, startY, translateX, translateY, endX, endY, duration) {
var timestamp = (new Date()).getTime() - startTime,
newX = parseInt(easingUtils.cubicOut(timestamp, startX, translateX, duration), 10),
newY = parseInt(easingUtils.cubicOut(timestamp, startY, translateY, duration), 10);
if (element.scrollLeft !== endX) {
element.scrollLeft = newX;
}
if (element.scrollTop !== endY) {
element.scrollTop = newY;
}
if ((newX !== endX || newY !== endY) &&
(newX >= 0 && newY >= 0) &&
state.currentTransition) {
util.requestAnimationFrame(state.currentTransition);
} else {
state.currentTransition = null;
}
}
// Translates scroll posotion directly or with an animation
// if duration is specified
// @param {Object} state Scrollview instance state
// @param {HTMLElement} element
// @param {number} x
// @param {number} y
// @param {number=} [duration]
function translate(state, element, x, y, duration) {
if (duration) {
state.currentTransition = translateTransition.bind(
null,
state,
element,
(new Date()).getTime(),
element.scrollLeft,
element.scrollTop,
x,
y,
element.scrollLeft + x,
element.scrollTop + y,
duration
);
util.requestAnimationFrame(state.currentTransition);
} else {
if (x) {
element.scrollLeft = element.scrollLeft + x;
}
if (y) {
element.scrollTop = element.scrollTop + y;
}
}
}
// Refresh jumpTop jumpLeft buttons
// @param {ns.widget.core.Scrollview} self
function repositionJumps(self) {
var ui = self._ui,
horizontalJumpButton = ui.jumpHorizontalButton,
verticalJumpButton = ui.jumpVerticalButton,
offsets = horizontalJumpButton || verticalJumpButton ? DOMUtils.getElementOffset(self.element) : null; // dont calc when not used
if (horizontalJumpButton) {
horizontalJumpButton.style.left = offsets.left + "px";
}
if (verticalJumpButton) {
verticalJumpButton.style.top = offsets.top + "px";
}
}
Scrollview.classes = classes;
Scrollview.prototype = new BaseWidget();
/**
* Builds the widget
* @param {HTMLElement} element
* @return {HTMLElement}
* @method _build
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype._build = function (element) {
//@TODO wrap element's content with external function
var self = this,
ui = self._ui,
view = selectors.getChildrenByClass(element, classes.view)[0] || document.createElement("div"),
clipStyle = element.style,
node = null,
child = element.firstChild,
options = self.options,
direction = options.scroll,
jumpButton,
jumpBackground;
view.className = classes.view;
while (child) {
node = child;
child = child.nextSibling;
if (view !== node) {
view.appendChild(node);
}
}
if (view.parentNode !== element) {
element.appendChild(view);
}
// setting view style
makePositioned(view);
element.classList.add(classes.clip);
switch (direction) {
case "x":
clipStyle.overflowX = "scroll";
break;
case "xy":
clipStyle.overflow = "scroll";
break;
default:
clipStyle.overflowY = "auto";
break;
}
if (options.scrollJump) {
if (direction.indexOf("x") > -1) {
jumpBackground = document.createElement("div");
jumpBackground.className = classes.jumpLeft;
jumpButton = document.createElement("div");
jumpBackground.appendChild(jumpButton);
element.appendChild(jumpBackground);
engine.instanceWidget(
jumpButton,
"Button",
{
"icon": "scrollleft",
"style": "box"
}
);
ui.jumpHorizontalButton = jumpBackground;
}
if (direction.indexOf("y") > -1) {
jumpBackground = document.createElement("div");
jumpBackground.className = classes.jumpTop;
jumpButton = document.createElement("div");
jumpBackground.appendChild(jumpButton);
element.appendChild(jumpBackground);
engine.instanceWidget(
jumpButton,
"Button",
{
"icon": "scrolltop",
"style": "box"
}
);
ui.jumpVerticalButton = jumpBackground;
}
}
ui.view = view;
// add scroll indicators
if (options.scrollIndicator) {
self._addOverflowIndicator(element);
}
return element;
};
/**
* Inits widget
* @method _init
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype._init = function (element) {
var ui = this._ui,
page = ui.page;
if (!ui.view) {
ui.view = selectors.getChildrenByClass(element, classes.view)[0];
}
if (!page) {
page = selectors.getClosestByClass(element, pageClass);
if (page) {
ui.page = page;
if (page.classList.contains(pageActiveClass) && this.options.scrollJump) {
repositionJumps(this);
}
}
}
};
/**
* Adds overflow indicators
* @param {HTMLElement} clip
* @method _addOverflowIndicator
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype._addOverflowIndicator = function (clip) {
clip.insertAdjacentHTML("beforeend",
"<div class='" + classes.indicatorTop + "'></div><div class='" + classes.indicatorBottom + "'></div>");
};
/**
* Clear classes and styles of indicators
* @param {HTMLElement} element
* @method clearIndicator
* @private
* @member ns.widget.core.Scrollview
*/
function clearIndicator (element) {
var clipClasses = element.classList,
topIndicator = selectors.getChildrenByClass(element, classes.indicatorTop)[0],
bottomIndicator = selectors.getChildrenByClass(element, classes.indicatorBottom)[0];
clipClasses.remove(classes.indicatorTopShown);
clipClasses.remove(classes.indicatorBottomShown);
clipClasses.remove(classes.indicatorRightShown);
clipClasses.remove(classes.indicatorLeftShown);
topIndicator.style = "";
bottomIndicator.style = "";
}
/**
* Set top and bottom indicators
* @param {HTMLElement} clip
* @param {object} options
* @method setTopAndBottomIndicators
* @private
* @member ns.widget.core.Scrollview
*/
function setTopAndBottomIndicators (clip, options) {
var topIndicator = selectors.getChildrenByClass(clip, classes.indicatorTop)[0],
bottomIndicator = selectors.getChildrenByClass(clip, classes.indicatorBottom)[0],
style;
// set top indicator
if (topIndicator) {
style = topIndicator.style;
style.width = options.width + "px";
style.top = options.clipTop + "px";
style.backgroundColor = options.color;
}
if (bottomIndicator) {
// set bottom indicator
style = bottomIndicator.style;
style.width = options.width + "px";
style.top = options.clipTop + options.clipHeight - DOMUtils.getElementHeight(bottomIndicator) + "px";
style.backgroundColor = options.color;
}
}
/**
* Show scroll indicators.
* @method _showScrollIndicator
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype._showScrollIndicator = function () {
var self = this,
clip = self.element,
view = self._ui.view,
scrollTop = clip.scrollTop,
clipHeight = DOMUtils.getElementHeight(clip),
clipOffset = DOMUtils.getElementOffset(clip),
viewHeight = DOMUtils.getElementHeight(view),
viewWidth = DOMUtils.getElementWidth(view),
viewOffset = DOMUtils.getElementOffset(view);
clearIndicator(clip);
switch (self.options.scroll) {
case "x":
// @todo
break;
case "xy":
// @todo
break;
default:
setTopAndBottomIndicators(clip, {
clipTop: clipOffset.top,
clipHeight: clipHeight,
width: viewWidth,
color: window.getComputedStyle(clip).backgroundColor
});
if (viewOffset.top - scrollTop < clipOffset.top) {
// the top is not visible
clip.classList.add(classes.indicatorTopShown);
}
if (viewOffset.top - scrollTop + viewHeight > clipOffset.top + clipHeight) {
// the bottom is not visible
clip.classList.add(classes.indicatorBottomShown);
}
}
};
/**
* Hide scroll indicators.
* @method _hideScrollIndicator
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype._hideScrollIndicator = function () {
var self = this,
timers = self._timers,
timer = timers.scrollIndicatorHide;
if (timer) {
window.clearTimeout(timer);
}
timers.scrollIndicatorHide = window.setTimeout(function () {
clearIndicator(self.element);
}, 1500);
};
/**
* Scrolls to specified position
*
* ### Example usage with TAU API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* </div>
* </div>
* <script>
* var scrollview = tau.widget.Scrollview(document.querySelector(".myPageClass > div[data-role=content]"));
* scrollview.scrollTo(0, 200, 1000); // scroll to 200px vertical with 1s animation
* </script>
*
* ### Example usage with jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* </div>
* </div>
* <script>
* var element = $(".myPageClass > div[data-role=content]"));
* element.scrollview();
* element.scrollview("scrollTo", 0, 200, 1000); // scroll to 200px vertical with 1s animation
* </script>
*
* @param {number} x
* @param {number} y
* @param {number=} [duration]
* @method scrollTo
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype.scrollTo = function (x, y, duration) {
var element = this.element;
this.translateTo(x - element.scrollLeft, y - element.scrollTop, duration);
};
/**
* Translates the scroll to specified position
*
* ### Example usage with TAU API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* </div>
* </div>
* <script>
* var scrollview = tau.widget.Scrollview(document.querySelector(".myPageClass > div[data-role=content]"));
* scrollview.translateTo(0, 200, 1000); // scroll forward 200px in vertical direction with 1s animation
* </script>
*
* ### Example usage with jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* </div>
* </div>
* <script>
* var element = $(".myPageClass > div[data-role=content]"));
* element.scrollview();
* element.scrollview("translateTo", 0, 200, 1000); // scroll forward 200px in vertical direction with 1s animation
* </script>
*
* @param {number} x
* @param {number} y
* @param {number=} [duration]
* @method translateTo
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype.translateTo = function (x, y, duration) {
translate(this.state, this.element, x, y, duration);
};
/**
* Ensures that specified element is visible in the
* clip area
*
* ### Example usage with TAU API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* <div class="testElementClass">somedata</div>
* </div>
* </div>
* <script>
* var scrollview = tau.widget.Scrollview(document.querySelector(".myPageClass > div[data-role=content]")),
* testElement = document.querySelector(".testElementClass");
* scrollview.ensureElementIsVisible(testelement);
* </script>
*
* ### Example usage with jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* <div class="testElementClass">somedata</div>
* </div>
* </div>
* <script>
* var element = $(".myPageClass > div[data-role=content]")),
* testElement = $(".testElementClass");
* element.scrollview();
* element.scrollview("ensureElementIsVisible", testElement);
* </script>
*
* @param {HTMLElement} element
* @method ensureElementIsVisible
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype.ensureElementIsVisible = function (element) {
var clip = this.element,
clipHeight = DOMUtils.getElementHeight(clip),
clipWidth = DOMUtils.getElementWidth(clip),
clipTop = 0,
clipBottom = clipHeight,
elementHeight = DOMUtils.getElementHeight(element),
elementWidth = DOMUtils.getElementWidth(element),
elementTop = 0,
elementBottom,
elementFits = clipHeight >= elementHeight && clipWidth >= elementWidth,
anchor,
anchorPositionX,
anchorPositionY,
parent,
findPositionAnchor = function (input) {
var id = input.getAttribute("id"),
tagName = input.tagName.toLowerCase();
if (id && ["input", "textarea", "button"].indexOf(tagName) > -1) {
return input.parentNode.querySelector("label[for=" + id + "]");
}
},
_true = true;
parent = element.parentNode;
while (parent && parent !== clip) {
elementTop += parent.offsetTop;
//elementLeft += parent.offsetLeft;
parent = parent.parentNode;
}
elementBottom = elementTop + elementHeight;
//elementRight = elementLeft + elementWidth;
switch (_true) {
case elementFits && clipTop < elementTop && clipBottom > elementBottom: // element fits in view is inside clip area
// pass, element position is ok
break;
case elementFits && clipTop < elementTop && clipBottom < elementBottom: // element fits in view but its visible only at top
case elementFits && clipTop > elementTop && clipBottom > elementBottom: // element fits in view but its visible only at bottom
case elementFits: // element fits in view but is not visible
this.centerToElement(element);
break;
case clipTop < elementTop && elementTop < clipBottom && clipBottom < elementBottom: // element visible only at top; eg. partly visible textarea
case clipTop > elementTop && clipBottom > elementBottom: // element visible only at bottom
// pass, we cant do anything, if we move the scroll
// the user could lost view of something he scrolled to
break;
default: // element is not visible
anchor = findPositionAnchor(element);
if (!anchor) {
anchor = element;
}
anchorPositionX = anchor.offsetLeft + DOMUtils.getCSSProperty(anchor, "margin-left", 0, "integer");
anchorPositionY = anchor.offsetTop + DOMUtils.getCSSProperty(anchor, "margin-top", 0, "integer");
parent = anchor.parentNode;
while (parent && parent !== clip) {
anchorPositionX += parent.offsetLeft;
anchorPositionY += parent.offsetTop;
parent = parent.parentNode;
}
this.scrollTo(anchorPositionX, anchorPositionY, this.scrollDuration);
break;
}
};
/**
* Centers specified element in the clip area
*
* ### Example usage with TAU API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* <div class="testElementClass">somedata</div>
* </div>
* </div>
* <script>
* var scrollview = tau.widget.Scrollview(document.querySelector(".myPageClass > div[data-role=content]")),
* testElement = document.querySelector(".testElementClass");
* scrollview.centerToElement(testelement);
* </script>
*
* ### Example usage with jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* <div class="testElementClass">somedata</div>
* </div>
* </div>
* <script>
* var element = $(".myPageClass > div[data-role=content]")),
* testElement = $(".testElementClass");
* element.scrollview();
* element.scrollview("centerToElement", testElement);
* </script>
*
* @param {HTMLElement} element
* @method centerToElement
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype.centerToElement = function (element) {
var clip = this.element,
deltaX = parseInt(DOMUtils.getElementWidth(clip) / 2 - DOMUtils.getElementWidth(element) / 2, 10),
deltaY = parseInt(DOMUtils.getElementHeight(clip) / 2 - DOMUtils.getElementHeight(element) / 2, 10),
elementPositionX = element.offsetLeft,
elementPositionY = element.offsetTop,
parent = element.parentNode;
while (parent && parent !== clip) {
elementPositionX += parent.offsetLeft + DOMUtils.getCSSProperty(parent, "margin-left", 0, "integer");
elementPositionY += parent.offsetTop + DOMUtils.getCSSProperty(parent, "margin-top", 0, "integer");
parent = parent.parentNode;
}
this.scrollTo(elementPositionX - deltaX, elementPositionY - deltaY, this.scrollDuration);
};
/**
* This is just for compatibility
* @method skipDragging
* @member ns.widget.core.Scrollview
* @deprecated 2.3
*/
Scrollview.prototype.skipDragging = function () {
if (ns.warn) {
ns.warn("ns.widget.core.Scrollview: skipDragging is deprecated");
}
}; // just for TWEBUIFW compat
/**
* Returns scroll current position
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* </div>
* </div>
* <script>
* var scrollview = tau.widget.Scrollview(document.querySelector(".myPageClass > div[data-role=content]")),
* currentPosition = scrollview.getScrollPosition();
* </script>
*
* ### Example usage with jQuery API
*
* @example
* <div class="myPageClass" data-role="page">
* <div data-role="content" data-scroll="y">
* content
* </div>
* </div>
* <script>
* var element = $(".myPageClass > div[data-role=content]")),
* position;
* element.scrollview();
* position = element.scrollview("getScrollPosition");
* </script>
*
* @return {Object}
* @method getScrollPosition
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype.getScrollPosition = function () {
var element = this.element;
return {
"x": element.scrollLeft,
"y": element.scrollTop
};
};
/**
* Binds scrollview events
* @method _bindEvents
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Scrollview
*/
Scrollview.prototype._bindEvents = function (element) {
var scrollTimer = null,
view = element.children[0],
lastClipHeight = DOMUtils.getElementHeight(element),
lastClipWidth = DOMUtils.getElementWidth(element),
notifyScrolled = function () {
eventUtils.trigger(element, "scrollstop");
window.clearTimeout(scrollTimer);
scrollTimer = null;
},
self = this,
//FIXME there should be some other way to get parent container
ui = self._ui,
page = ui.page,
jumpTop = ui.jumpVerticalButton,
jumpLeft = ui.jumpHorizontalButton,
repositionJumpsCallback,
jumpTopCallback,
jumpLeftCallback,
callbacks = self._callbacks;
if (page) {
if (this.options.scrollJump) {
repositionJumpsCallback = repositionJumps.bind(null, this);
jumpTopCallback = function () {
self.scrollTo(element.scrollLeft, 0, 250);
};
jumpLeftCallback = function () {
self.scrollTo(0, element.scrollTop, 250);
};
page.addEventListener("pageshow", repositionJumpsCallback, false);
if (jumpTop) {
jumpTop.firstChild.addEventListener("vclick", jumpTopCallback, false);
}
if (jumpLeft) {
jumpLeft.firstChild.addEventListener("vclick", jumpLeftCallback, false);
}
callbacks.repositionJumps = repositionJumpsCallback;
callbacks.jumpTop = jumpTopCallback;
callbacks.jumpLeft = jumpLeftCallback;
}
element.addEventListener("scroll", function () {
if (scrollTimer) {
window.clearTimeout(scrollTimer);
} else {
eventUtils.trigger(element, "scrollstart");
}
scrollTimer = window.setTimeout(notifyScrolled, 100);
eventUtils.trigger(element, "scrollupdate");
}, false);
document.addEventListener("vmousedown", function () {
if (currentTransition) {
currentTransition = null;
}
}, false);
if (self.options.scrollIndicator) {
callbacks.scrollUpdate = self._showScrollIndicator.bind(self);
element.addEventListener("scrollupdate", callbacks.scrollUpdate , false);
callbacks.scrollStop = self._hideScrollIndicator.bind(self);
element.addEventListener("scrollstop", callbacks.scrollStop , false);
}
}
};
Scrollview.prototype._destroy = function () {
var self = this,
element = self.element,
ui = self._ui,
page = ui.page,
scrollJump = this.options.scrollJump,
jumpTop = ui.jumpVerticalButton,
jumpLeft = ui.jumpHorizontalButton,
callbacks = self._callbacks,
repositionJumpsCallback = callbacks.repositionJumps,
jumpTopCallback = callbacks.jumpTop,
jumpLeftCallback = callbacks.jumpLeft;
if (scrollJump) {
if (page && repositionJumpsCallback) {
page.removeEventListener("pageshow", repositionJumpsCallback, false);
}
if (jumpTop && jumpTopCallback) {
jumpTop.firstChild.removeEventListener("vclick", jumpTopCallback, false);
}
if (jumpLeft && jumpLeftCallback) {
jumpLeft.firstChild.removeEventListener("vclick", jumpLeftCallback, false);
}
}
if (self.options.scrollIndicator) {
element.removeEventListener("scrollupdate", callbacks.scrollUpdate , false);
}
if (self._timers.scrollIndicatorHide) {
window.clearTimeout(self._timers.scrollIndicatorHide);
}
};
ns.widget.core.Scrollview = Scrollview;
}(window, window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* @class ns.widget.mobile.Tab
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
events = ns.event,
Tab = function () {
},
/**
* Object with class dictionary
* @property {Object} classes
* @static
* @member ns.widget.mobile.Tab
* @readonly
*/
classes = {
},
CustomEvent = {
TAB_CHANGE: "tabchange"
},
prototype = new BaseWidget();
Tab.prototype = prototype;
Tab.classes = classes;
/**
* Set the active tab
* @method setActive
* @param {number} index of the tab
* @public
* @member ns.widget.mobile.Tab
*/
prototype._setActive = function(index) {
var element = this.element;
events.trigger(element, CustomEvent.TAB_CHANGE, {
active: index
});
};
/**
* Set the active tab
* @method setActive
* @param {number} index of the tab
* @public
* @member ns.widget.mobile.Tab
*/
prototype.setActive = function(index) {
this._setActive(index);
};
/**
* Get the active tab
* @method setActive
* @param {number} index of the tab
* @public
* @member ns.widget.mobile.Tab
*/
prototype._getActive = function() {
return this.options.active;
};
/**
* Get the active tab
* @method setActive
* @param {number} index of the tab
* @public
* @member ns.widget.mobile.Tab
*/
prototype.getActive = function() {
return this._getActive();
};
ns.widget.core.Tab = Tab;
engine.defineWidget(
"Tab",
"",
["setActive", "getActive"],
Tab,
"tizen"
);
}(window.document, ns));
/*global window, define, Event, console */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #TabIndicator Widget
* Widget create tabs indicator.
* @class ns.widget.core.TabIndicator
* @since 2.3
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var Tab = ns.widget.core.Tab,
engine = ns.engine,
object = ns.util.object,
TabIndicator = function() {
this.tabSize = 0;
this.width = 0;
},
TabPrototype = Tab.prototype,
prototype = new Tab();
TabIndicator.prototype = prototype;
prototype._init = function(element) {
var o = this.options;
this.width = element.offsetWidth;
element.classList.add( o.wrapperClass );
};
prototype._configure = function( ) {
/**
* @property {Object} options Options for widget
* @property {number} [options.margin=2]
* @property {boolean} [options.triggerEvent=false]
* @property {string} [options.wrapperClass="ui-tab-indicator]
* @property {string} [options.itemClass="ui-tab-item"]
* @property {string} [options.activeClass="ui-tab-active"]
* @member ns.widget.core.TabIndicator
*/
object.merge(this.options, {
margin: 4,
triggerEvent: false,
wrapperClass: "ui-tab-indicator",
itemClass: "ui-tab-item",
activeClass: "ui-tab-active",
active: 0
});
};
prototype._createIndicator = function() {
var o = this.options,
wrap = document.createDocumentFragment(),
widthTable = [],
margin = o.margin,
i = 0,
len = this.tabSize,
width = this.width-margin*(len-1),
std = Math.floor(width / len),
remain = width % len,
span, offset=0;
for (i=0; i < len; i++) {
widthTable[i] = std;
}
for ( i= Math.floor((len-remain)/2); remain > 0; i++, remain-- ) {
widthTable[i] += 1;
}
for (i=0; i < len; i++) {
span = document.createElement("span");
span.classList.add( o.itemClass );
span.style.width = widthTable[i] + "px";
span.style.left = offset + "px";
offset += widthTable[i] + margin;
if ( i === o.active ) {
span.classList.add( o.activeClass );
}
wrap.appendChild(span);
}
this.element.appendChild( wrap );
};
prototype._removeIndicator = function() {
this.element.innerHTML = "";
};
prototype._fireEvent = function(eventName, detail) {
ns.fireEvent( this.element, eventName, detail );
};
prototype._refresh = function() {
this._removeIndicator();
this._createIndicator();
};
/**
* @method setActive
* @param index
* @member ns.widget.core.TabIndicator
*/
prototype._setActive = function ( index ) {
var o = this.options,
nodes = this.element.children;
o.active = index;
[].forEach.call(nodes, function( element ) {
element.classList.remove( o.activeClass );
});
if ( index < nodes.length ) {
nodes[index].classList.add( o.activeClass );
TabPrototype._setActive.call(this, index);
}
};
/**
* @method setSize
* @param size
* @member ns.widget.core.TabIndicator
*/
prototype.setSize = function( size ) {
var needRefresh = this.tabSize !== size;
this.tabSize = size;
if ( needRefresh ) {
this.refresh();
}
};
prototype._destroy = function() {
var o = this.options;
this._removeIndicator();
this.element.classList.remove( o.wrapperClass );
};
ns.widget.core.TabIndicator = TabIndicator;
engine.defineWidget(
"TabIndicator",
".ui-tab",
["setActive", "getActive", "setSize"],
TabIndicator
);
}(window.document, ns));
/*global window, define, ns*/
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # SectionChanger Widget
* Shows a control that you can use to scroll through multiple *section*
* elements.
*
* The section changer widget provides an application architecture, which has
* multiple sections on a page and enables scrolling through the *section* elements.
*
* ## Manual constructor
*
* @example
* <div id="hasSectionchangerPage" class="ui-page">
* <header class="ui-header">
* <h2 class="ui-title">SectionChanger</h2>
* </header>
* <div id="sectionchanger" class="ui-content">
* <!--Section changer has only one child-->
* <div>
* <section>
* <h3>LEFT1 PAGE</h3>
* </section>
* <section class="ui-section-active">
* <h3>MAIN PAGE</h3>
* </section>
* <section>
* <h3>RIGHT1 PAGE</h3>
* </section>
* </div>
* </div>
* </div>
* <script>
* (function () {
* var page = document.getElementById("hasSectionchangerPage"),
* element = document.getElementById("sectionchanger"),
* sectionChanger;
*
* page.addEventListener("pageshow", function () {
* // Create the SectionChanger object
* sectionChanger = new tau.SectionChanger(element, {
* circular: true,
* orientation: "horizontal",
* useBouncingEffect: true
* });
* });
*
* page.addEventListener("pagehide", function () {
* // Release the object
* sectionChanger.destroy();
* });
* })();
* </script>
*
* ## Handling Events
*
* To handle section changer events, use the following code:
*
* @example
* <script>
* (function () {
* var changer = document.getElementById("sectionchanger");
* changer.addEventListener("sectionchange", function (event) {
* console.debug(event.detail.active + " section is active.");
* });
* })();
* </script>
*
* @class ns.widget.core.SectionChanger
* @since 2.2
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var Scroller = ns.widget.core.scroller.Scroller,
Gesture = ns.event.gesture,
engine = ns.engine,
utilsObject = ns.util.object,
utilsEvents = ns.event,
eventType = ns.util.object.merge({
/**
* Triggered when the section is changed.
* @event sectionchange
* @member ns.widget.core.SectionChanger
*/
CHANGE: "sectionchange"
}, Scroller.EventType),
classes = {
uiSectionChanger: "ui-section-changer"
};
function SectionChanger() {
this.options = {};
}
function calculateCustomLayout(direction, elements, lastIndex) {
var len = lastIndex !== undefined ? lastIndex : elements.length,
result = 0,
i;
for (i = 0; i < len; i++) {
result += direction === Scroller.Orientation.HORIZONTAL ? elements[i].offsetWidth : elements[i].offsetHeight;
}
return result;
}
function calculateCenter(direction, elements, index) {
var result = calculateCustomLayout(direction, elements, index + 1);
result -= direction === Scroller.Orientation.HORIZONTAL ? elements[index].offsetWidth / 2 : elements[index].offsetHeight / 2;
return result;
}
utilsObject.inherit(SectionChanger, Scroller, {
_build: function (element) {
this.tabIndicatorElement = null;
this.tabIndicator = null;
this.sections = null;
this.sectionPositions = [];
this.activeIndex = 0;
this.beforeIndex = 0;
this._super(element);
element.classList.add(classes.uiSectionChanger);
return element;
},
_configure : function () {
this._super();
/**
* Options for widget
* @property {Object} options
* @property {"horizontal"|"vertical"} [options.orientation="horizontal"] Sets the section changer orientation:
* @property {boolean} [options.circular=false] Presents the sections in a circular scroll fashion.
* @property {boolean} [options.useBouncingEffect=false] Shows a scroll end effect on the scroll edge.
* @property {string} [options.items="section"] Defines the section element selector.
* @property {string} [options.activeClass="ui-section-active"] Specifies the CSS classes which define the active section element. Add the specified class (ui-section-active) to a *section* element to indicate which section must be shown first. By default, the first section is shown first.
* @property {boolean} [options.fillContent=true] declare to section tag width to fill content or not.
* @member ns.widget.core.SectionChanger
*/
this.options = utilsObject.merge(this.options, {
items: "section",
activeClass: "ui-section-active",
circular: false,
animate: true,
animateDuration: 100,
orientation: "horizontal",
changeThreshold: -1,
useTab: false,
fillContent: true
});
},
_init: function (element) {
var o = this.options,
scroller = this.scroller,
offsetHeight,
sectionLength, i, className;
scroller.style.position = "absolute";
offsetHeight = element.offsetHeight;
if (offsetHeight === 0) {
offsetHeight = element.parentNode.offsetHeight;
element.style.height = offsetHeight + "px";
}
this._sectionChangerWidth = element.offsetWidth;
this._sectionChangerHeight = offsetHeight;
this._sectionChangerHalfWidth = this._sectionChangerWidth / 2;
this._sectionChangerHalfHeight = this._sectionChangerHeight / 2;
this.orientation = o.orientation === "horizontal" ? Scroller.Orientation.HORIZONTAL : Scroller.Orientation.VERTICAL;
if (o.scrollbar === "tab") {
o.scrollbar = false;
o.useTab = true;
}
this.sections = typeof o.items === "string" ?
scroller.querySelectorAll(o.items) :
o.items;
sectionLength = this.sections.length;
if (o.circular && sectionLength < 3) {
throw "if you use circular option, you must have at least three sections.";
}
if (this.activeIndex >= sectionLength) {
this.activeIndex = sectionLength - 1;
}
for (i = 0; i < sectionLength; i++) {
className = this.sections[i].className;
if (className && className.indexOf(o.activeClass) > -1) {
this.activeIndex = i;
}
this.sectionPositions[i] = i;
}
this._prepareLayout();
this._initLayout();
this._super();
this._repositionSections(true);
this.setActiveSection(this.activeIndex);
// set corret options values.
if (!o.animate) {
o.animateDuration = 0;
}
if (o.changeThreshold < 0) {
o.changeThreshold = this._sectionChangerHalfWidth;
}
return element;
},
_prepareLayout: function () {
var o = this.options,
sectionLength = this.sections.length,
width = this._sectionChangerWidth,
height = this._sectionChangerHeight,
orientation = this.orientation,
scrollerStyle = this.scroller.style,
tabHeight;
if (o.useTab) {
this._initTabIndicator();
tabHeight = this.tabIndicatorElement.offsetHeight;
height -= tabHeight;
this._sectionChangerHalfHeight = height / 2;
this.element.style.height = height + "px";
this._sectionChangerHeight = height;
}
if (orientation === Scroller.Orientation.HORIZONTAL) {
scrollerStyle.width = (o.fillContent ? width * sectionLength : calculateCustomLayout(orientation, this.sections)) + "px";
scrollerStyle.height = height + "px"; //set Scroller width
} else {
scrollerStyle.width = width + "px"; //set Scroller width
scrollerStyle.height = (o.fillContent ? height * sectionLength : calculateCustomLayout(orientation, this.sections)) + "px";
}
},
_initLayout: function () {
var sectionStyle = this.sections.style,
left = 0,
top = 0,
i, sectionLength;
//section element has absolute position
for (i = 0, sectionLength = this.sections.length; i < sectionLength; i++) {
//Each section set initialize left position
sectionStyle = this.sections[i].style;
sectionStyle.position = "absolute";
if (this.options.fillContent) {
sectionStyle.width = this._sectionChangerWidth + "px";
sectionStyle.height = this._sectionChangerHeight + "px";
}
if (this.orientation === Scroller.Orientation.HORIZONTAL) {
top = 0;
left = calculateCustomLayout(this.orientation, this.sections, i);
} else {
top = calculateCustomLayout(this.orientation, this.sections, i);
left = 0;
}
sectionStyle.top = top + "px";
sectionStyle.left = left + "px";
}
},
_initBouncingEffect: function () {
var o = this.options;
if (!o.circular) {
this._super();
}
},
_translateScrollbar: function (x, y, duration, autoHidden) {
var offset;
if (!this.scrollbar) {
return;
}
if (this.orientation === Scroller.Orientation.HORIZONTAL) {
offset = (-x + this.minScrollX);
} else {
offset = (-y + this.minScrollY);
}
this.scrollbar.translate(offset, duration, autoHidden);
},
_translateScrollbarWithPageIndex: function (pageIndex, duration) {
var offset;
if (!this.scrollbar) {
return;
}
offset = calculateCustomLayout(this.orientation, this.sections, this.activeIndex);
this.scrollbar.translate(offset, duration);
},
_initTabIndicator: function () {
var elem = this.tabIndicatorElement = document.createElement("div");
this.element.parentNode.insertBefore(elem, this.element);
this.tabIndicator = new engine.instanceWidget(elem, "TabIndicator");
this.tabIndicator.setSize(this.sections.length);
this.tabIndicator.setActive(this.activeIndex);
this.tabIndicatorHandler = function (e) {
this.tabIndicator.setActive(e.detail.active);
}.bind(this);
this.element.addEventListener(eventType.CHANGE, this.tabIndicatorHandler, false);
},
_clearTabIndicator: function () {
if (this.tabIndicator) {
this.element.parentNode.removeChild(this.tabIndicatorElement);
this.element.removeEventListener(eventType.CHANGE, this.tabIndicatorHandler, false);
this.tabIndicator.destroy();
this.tabIndicator = null;
this.tabIndicatorElement = null;
this.tabIndicatorHandler = null;
}
},
_resetLayout: function () {
var //scrollerStyle = this.scroller.style,
sectionStyle = this.sections.style,
i, sectionLength;
//scrollerStyle.width = "";
//scrollerStyle.height = "";
//this.scroller || this.scroller._resetLayout();
for (i = 0, sectionLength = this.sections.length; i < sectionLength; i++) {
sectionStyle = this.sections[i].style;
sectionStyle.position = "";
sectionStyle.width = "";
sectionStyle.height = "";
sectionStyle.top = "";
sectionStyle.left = "";
}
this._super();
},
_bindEvents: function () {
this._super();
ns.event.enableGesture(
this.scroller,
new ns.event.gesture.Swipe({
orientation: this.orientation === Scroller.Orientation.HORIZONTAL ?
Gesture.Orientation.HORIZONTAL :
Gesture.Orientation.VERTICAL
})
);
utilsEvents.on(this.scroller,
"swipe transitionEnd webkitTransitionEnd mozTransitionEnd msTransitionEnd oTransitionEnd", this);
},
_unbindEvents: function () {
this._super();
if (this.scroller) {
ns.event.disableGesture(this.scroller);
utilsEvents.off(this.scroller,
"swipe transitionEnd webkitTransitionEnd mozTransitionEnd msTransitionEnd oTransitionEnd", this);
}
},
/**
* This method manages events.
* @method handleEvent
* @returns {Event} event
* @member ns.widget.core.SectionChanger
*/
handleEvent: function (event) {
this._super(event);
switch (event.type) {
case "swipe":
this._swipe(event);
break;
case "webkitTransitionEnd":
case "mozTransitionEnd":
case "msTransitionEnd":
case "oTransitionEnd":
case "transitionEnd":
if (event.target === this.scroller) {
this._endScroll();
}
break;
}
},
_notifyChanagedSection: function (index) {
var activeClass = this.options.activeClass,
sectionLength = this.sections.length,
i=0, section;
for (i=0; i < sectionLength; i++) {
section = this.sections[i];
section.classList.remove(activeClass);
if (i === this.activeIndex) {
section.classList.add(activeClass);
}
}
this._fireEvent(eventType.CHANGE, {
active: index
});
},
/**
* Changes the currently active section element.
* @method setActiveSection
* @param {number} index
* @param {number} duration For smooth scrolling,
* the duration parameter must be in milliseconds.
* @member ns.widget.core.SectionChanger
*/
setActiveSection: function (index, duration, direct) {
var position = this.sectionPositions[ index ],
scrollbarDuration = duration,
oldActiveIndex = this.activeIndex,
newX=0,
newY= 0,
centerX = 0,
centerY = 0;
if (this.orientation === Scroller.Orientation.HORIZONTAL) {
newX = this._sectionChangerHalfWidth - calculateCenter(this.orientation, this.sections, position);
} else {
newY = this._sectionChangerHalfHeight - calculateCenter(this.orientation, this.sections, position);
}
if (this.beforeIndex - index > 1 || this.beforeIndex - index < -1) {
scrollbarDuration = 0;
}
this.activeIndex = index;
this.beforeIndex = this.activeIndex;
if (newX !== this.scrollerOffsetX || newY !== this.scrollerOffsetY) {
if (direct !== false) {
this._fireEvent( eventType.START );
this.scrolled = true;
}
this._translate(newX, newY, duration);
this._translateScrollbarWithPageIndex(index, scrollbarDuration);
} else {
this._endScroll();
}
// notify changed section.
if (this.activeIndex !== oldActiveIndex) {
this._notifyChanagedSection(this.activeIndex);
}
},
/**
* Gets the currently active section element's index.
* @method getActiveSectionIndex
* @returns {number}
* @member ns.widget.core.SectionChanger
*/
getActiveSectionIndex: function () {
return this.activeIndex;
},
_start: function (e) {
this._super(e);
this.beforeIndex = this.activeIndex;
},
_move: function (e) {
var changeThreshold = this.options.changeThreshold,
delta = this.orientation === Scroller.Orientation.HORIZONTAL ? e.detail.deltaX : e.detail.deltaY,
oldActiveIndex = this.activeIndex;
this._super(e);
if (!this.scrolled) {
return;
}
if (delta > changeThreshold) {
this.activeIndex = this._calculateIndex(this.beforeIndex - 1);
} else if (delta < -changeThreshold) {
this.activeIndex = this._calculateIndex(this.beforeIndex + 1);
} else {
this.activeIndex = this.beforeIndex;
}
// notify changed section.
if (this.activeIndex !== oldActiveIndex) {
this._notifyChanagedSection(this.activeIndex);
}
},
_end: function (/* e */) {
if ( this.scrollbar ) {
this.scrollbar.end();
}
if (!this.enabled || this.scrollCanceled || !this.dragging) {
return;
}
// bouncing effect
if (this.bouncingEffect) {
this.bouncingEffect.dragEnd();
}
this.setActiveSection(this.activeIndex, this.options.animateDuration, false);
this.dragging = false;
},
_swipe: function (e) {
var offset = e.detail.direction === Gesture.Direction.UP || e.detail.direction === Gesture.Direction.LEFT ? 1 : -1,
newIndex = this._calculateIndex(this.beforeIndex + offset);
if (!this.enabled || this.scrollCanceled || !this.dragging) {
return;
}
// bouncing effect
if (this.bouncingEffect) {
this.bouncingEffect.dragEnd();
}
if (this.activeIndex !== newIndex) {
this.activeIndex = newIndex;
this._notifyChanagedSection(newIndex);
}
this.setActiveSection(newIndex, this.options.animateDuration, false);
this.dragging = false;
},
_endScroll: function () {
if (!this.enabled || !this.scrolled || this.scrollCanceled) {
return;
}
this._repositionSections();
this._super();
},
_repositionSections: function (init) {
// if developer set circular option is true, this method used when webkitTransitionEnd event fired
var sectionLength = this.sections.length,
curPosition = this.sectionPositions[this.activeIndex],
centerPosition = window.parseInt(sectionLength/2, 10),
circular = this.options.circular,
centerX = 0,
centerY = 0,
i, sectionStyle, sIdx, top, left, newX, newY;
if (this.orientation === Scroller.Orientation.HORIZONTAL) {
newX = -(calculateCenter(this.orientation, this.sections, (circular ? centerPosition : this.activeIndex)));
newY = 0;
} else {
newX = 0;
newY = -(calculateCenter(this.orientation, this.sections, (circular ? centerPosition : this.activeIndex)));
}
this._translateScrollbarWithPageIndex(this.activeIndex);
if (init || (curPosition === 0 || curPosition === sectionLength - 1)) {
if (this.orientation === Scroller.Orientation.HORIZONTAL) {
centerX = this._sectionChangerHalfWidth + newX;
} else {
centerY = this._sectionChangerHalfHeight + newY;
}
this._translate(centerX, centerY);
if (circular) {
for (i = 0; i < sectionLength; i++) {
sIdx = (sectionLength + this.activeIndex - centerPosition + i) % sectionLength;
sectionStyle = this.sections[ sIdx ].style;
this.sectionPositions[sIdx] = i;
if (this.orientation === Scroller.Orientation.HORIZONTAL) {
top = 0;
left = calculateCustomLayout(this.orientation, this.sections, i);
} else {
top = calculateCustomLayout(this.orientation, this.sections, i);
left = 0;
}
sectionStyle.top = top + "px";
sectionStyle.left = left + "px";
}
}
}
},
_calculateIndex: function (newIndex) {
var sectionLength = this.sections.length;
if (this.options.circular) {
newIndex = (sectionLength + newIndex) % sectionLength;
} else {
newIndex = newIndex < 0 ? 0 : (newIndex > sectionLength - 1 ? sectionLength - 1 : newIndex);
}
return newIndex;
},
_clear: function () {
this._clearTabIndicator();
this._super();
this.sectionPositions.length = 0;
}
});
ns.widget.core.SectionChanger = SectionChanger;
engine.defineWidget(
"SectionChanger",
"[data-role='section-changer'], .ui-section-changer",
["getActiveSectionIndex", "setActiveSection"],
SectionChanger
);
}(window.document, ns));
/*global window, define, NodeList, HTMLCollection */
/*jslint plusplus: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @author Jadwiga Sosnowska <j.sosnowska@partner.samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @author Maciej Moczulski <m.moczulski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/
(function (window, document, ns) {
"use strict";
/**
* @property {DocumentFragment} fragment
* @member ns.util.DOM
* @private
* @static
*/
/*
* @todo maybe can be moved to function scope?
*/
var fragment = document.createDocumentFragment(),
/**
* @property {DocumentFragment} fragment2
* @member ns.util.DOM
* @private
* @static
*/
/*
* @todo maybe can be moved to function scope?
*/
fragment2 = document.createDocumentFragment(),
/**
* @property {number} [containerCounter=0]
* @member ns.util.DOM
* @private
* @static
*/
/*
* @todo maybe can be moved to function scope?
*/
containerCounter = 0,
/**
* Alias to Array.slice method
* @method slice
* @member ns.util.DOM
* @private
* @static
*/
slice = [].slice,
DOM = ns.util.DOM;
/**
* Appends node or array-like node list array to context
* @method appendNodes
* @member ns.util.DOM
* @param {HTMLElement} context
* @param {HTMLElement|HTMLCollection|NodeList|Array} elements
* @return {HTMLElement|Array|null}
* @static
* @throws {string}
*/
DOM.appendNodes = function (context, elements) {
var i,
len;
if (context) {
if (elements instanceof Array || elements instanceof NodeList || elements instanceof HTMLCollection) {
elements = slice.call(elements);
for (i = 0, len = elements.length; i < len; ++i) {
context.appendChild(elements[i]);
}
} else {
context.appendChild(elements);
}
return elements;
}
throw "Context empty!";
};
/**
* Replaces context with node or array-like node list
* @method replaceWithNodes
* @member ns.util.DOM
* @param {HTMLElement} context
* @param {HTMLElement|HTMLCollection|NodeList|Array} elements
* @return {HTMLElement|Array|null}
* @static
*/
DOM.replaceWithNodes = function (context, elements) {
if (elements instanceof Array || elements instanceof NodeList || elements instanceof HTMLCollection) {
elements = this.insertNodesBefore(context, elements);
context.parentNode.removeChild(context);
} else {
context.parentNode.replaceChild(elements, context);
}
return elements;
};
/**
* Remove all children
* @method removeAllChildren
* @member ns.util.DOM
* @param {HTMLElement} context
* @static
*/
DOM.removeAllChildren = function (context) {
context.innerHTML = "";
};
/**
* Inserts node or array-like node list before context
* @method insertNodesBefore
* @member ns.util.DOM
* @param {HTMLElement} context
* @param {HTMLElement|HTMLCollection|NodeList|Array} elements
* @return {HTMLElement|Array|null}
* @static
* @throws {string}
*/
DOM.insertNodesBefore = function (context, elements) {
var i,
len,
parent;
if (context) {
parent = context.parentNode;
if (elements instanceof Array || elements instanceof NodeList || elements instanceof HTMLCollection) {
elements = slice.call(elements);
for (i = 0, len = elements.length; i < len; ++i) {
parent.insertBefore(elements[i], context);
}
} else {
parent.insertBefore(elements, context);
}
return elements;
}
throw "Context empty!";
};
/**
* Inserts node after context
* @method insertNodeAfter
* @member ns.util.DOM
* @param {HTMLElement} context
* @param {HTMLElement} element
* @return {HTMLElement}
* @static
* @throws {string}
*/
DOM.insertNodeAfter = function (context, element) {
if (context) {
context.parentNode.insertBefore(element, context.nextSibling);
return element;
}
throw "Context empty!";
};
/**
* Wraps element or array-like node list in html markup
* @method wrapInHTML
* @param {HTMLElement|NodeList|HTMLCollection|Array} elements
* @param {string} html
* @return {HTMLElement|NodeList|Array} wrapped element
* @member ns.util.DOM
* @static
*/
DOM.wrapInHTML = function (elements, html) {
var container = document.createElement("div"),
contentFlag = false,
elementsLen = elements.length,
//if elements is nodeList, retrieve parentNode of first node
originalParentNode = elementsLen ? elements[0].parentNode : elements.parentNode,
next = elementsLen ? elements[elementsLen - 1].nextSibling : elements.nextSibling,
innerContainer;
fragment.appendChild(container);
html = html.replace(/(\$\{content\})/gi, function () {
contentFlag = true;
return "<span id='temp-container-" + (++containerCounter) + "'></span>";
});
container.innerHTML = html;
if (contentFlag === true) {
innerContainer = container.querySelector("span#temp-container-" + containerCounter);
elements = this.replaceWithNodes(innerContainer, elements);
} else {
innerContainer = container.children[0];
elements = this.appendNodes(innerContainer || container, elements);
}
// move the nodes
while (fragment.firstChild.firstChild) {
fragment2.appendChild(fragment.firstChild.firstChild);
}
// clean up
while (fragment.firstChild) {
fragment.removeChild(fragment.firstChild);
}
if (originalParentNode) {
if (next) {
originalParentNode.insertBefore(fragment2, next);
} else {
originalParentNode.appendChild(fragment2);
}
} else {
// clean up
while (fragment2.firstChild) {
fragment2.removeChild(fragment2.firstChild);
}
}
return elements;
};
}(window, window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Wearable Widget Reference
* The Tizen Web UI service provides rich Tizen widgets that are optimized for the Tizen Web browser. You can use the widgets for:
*
* - CSS animation
* - Rendering
*
* The following table displays the widgets provided by the Tizen Web UI service.
* @class ns.widget.wearable
* @seeMore https://developer.tizen.org/dev-guide/2.2.1/org.tizen.web.uiwidget.apireference/html/web_ui_framework.htm "Web UI Framework Reference"
* @author Maciej Urbanski <m.urbanski@samsung.com>
*/
(function (window, ns) {
"use strict";
ns.widget.wearable = ns.widget.wearable || {};
}(window, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Page Widget
* Page is main element of application's structure.
*
* ## Default selectors
* In the Tizen Web UI framework the application page structure is based on a header, content and footer elements:
*
* - **The header** is placed at the top, and displays the page title and optionally buttons.
* - **The content** is the section below the header, showing the main content of the page.
* - **The footer** is a bottom part of page which can display for example buttons
*
* The following table describes the specific information for each section.
*
* <table>
* <tr>
* <th>Section</th>
* <th>Class</th>
* <th>Mandatory</th>
* <th>Description</th>
* </tr>
* <tr>
* <td rowspan="2">Page</td>
* <td>ui-page</td>
* <td>Yes</td>
* <td>Defines the element as a page.
*
* The page widget is used to manage a single item in a page-based architecture.
*
* A page is composed of header (optional), content (mandatory), and footer (optional) elements.</td>
* </tr>
* <tr>
* <td>ui-page-active</td>
* <td>No</td>
* <td>If an application has a static start page, insert the ui-page-active class in the page element to speed up the application launch. The start page with the ui-page-active class can be displayed before the framework is fully loaded.
*
*If this class is not used, the framework inserts the class automatically to the first page of the application. However, this has a slowing effect on the application launch, because the page is displayed only after the framework is fully loaded.</td>
* </tr>
* <tr>
* <td>Header</td>
* <td>ui-header</td>
* <td>No</td>
* <td>Defines the element as a header.</td>
* </tr>
* <tr>
* <td>Content</td>
* <td>ui-content</td>
* <td>Yes</td>
* <td>Defines the element as content.</td>
* </tr>
* <tr>
* <td>Footer</td>
* <td>ui-footer</td>
* <td>No</td>
* <td>Defines the element as a footer.
*
* The footer section is mostly used to include option buttons.</td>
* </tr>
* </table>
*
* All elements with class=ui-page will be become page widgets
*
* @example
* <!--Page layout-->
* <div class="ui-page ui-page-active">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <footer class="ui-footer"></footer>
* </div>
*
* <!--Page layout with more button in header-->
* <div class="ui-page ui-page-active">
* <header class="ui-header ui-has-more">
* <h2 class="ui-title">Call menu</h2>
* <button type="button" class="ui-more ui-icon-overflow">More Options</button>
* </header>
* <div class="ui-content">Content message</div>
* <footer class="ui-footer">
* <button type="button" class="ui-btn">Footer Button</button>
* </footer>
* </div>
*
* ## Manual constructor
* For manual creation of page widget you can use constructor of widget from **tau** namespace:
*
* @example
* var pageElement = document.getElementById("page"),
* page = tau.widget.page(buttonElement);
*
* Constructor has one require parameter **element** which are base **HTMLElement** to create widget. We recommend get this element by method *document.getElementById*.
*
* ## Multi-page Layout
*
* You can implement a template containing multiple page containers in the application index.html file.
*
* In the multi-page layout, the main page is defined with the ui-page-active class. If no page has the ui-page-active class, the framework automatically sets up the first page in the source order as the main page. You can improve the launch performance by explicitly defining the main page to be displayed first. If the application has to wait for the framework to set up the main page, the page is displayed with some delay only after the framework is fully loaded.
*
* You can link to internal pages by referring to the ID of the page. For example, to link to the page with an ID of two, the link element needs the href="#two" attribute in the code, as in the following example.
*
* @example
* <!--Main page-->
* <div id="one" class="ui-page ui-page-active">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <footer class="ui-footer"></footer>
* </div>
*
* <!--Secondary page-->
* <div id="two" class="ui-page">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <footer class="ui-footer"></footer>
* </div>
*
* To find the currently active page, use the ui-page-active class.
*
* ## Changing Pages
* ### Go to page in JavaScript
* To change page use method *tau.changePage*
*
* @example
* tau.changePage("page-two");
*
* ### Back in JavaScript
* To back to previous page use method *tau.back*
*
* @example
* tau.back();
*
* ## Transitions
*
* When changing the active page, you can use a page transition.
*
* Tizen Web UI Framework does not apply transitions by default. To set a custom transition effect, you must add the data-transition attribute to a link:
*
* @example
* <a href="index.html" data-transition="slideup">I'll slide up</a>
*
* To set a default custom transition effect for all pages, use the pageTransition property:
*
* @example
* tau.defaults.pageTransition = "slideup";
*
* ### Transitions list
*
* - **none** no transition.
* - **slideup** Makes the content of the next page slide up, appearing to conceal the content of the previous page.
*
* ## Handling Page Events
*
* With page widget we have connected many of events.
*
* To handle page events, use the following code:
*
* @example
* <div id="page" class="ui-page">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* </div>
*
* <script>
* var page = document.getElementById("page");
* page.addEventListener("Event", function(event) {
* // Your code
* });
* </script>
*
* To bind an event callback on the Back key, use the following code:
*
* Full list of available events is in [events list section](#events-list).
*
* To bind an event callback on the Back key, use the following code:
*
* @example
* <script>
* window.addEventListener("tizenhwkey", function (event) {
* if (event.keyName == "back") {
* // Call window.history.back() to go to previous browser window
* // Call tizen.application.getCurrentApplication().exit() to exit application
* // Add script to add another behavior
* }
* });
* </script>
*
* ## Options for Page Widget
*
* Page widget hasn't any options.
*
* ## Methods
*
* To call method on widget you can use tau API:
*
* @example
* var pageElement = document.getElementById("page"),
* page = tau.widget.page(buttonElement);
*
* page.methodName(methodArgument1, methodArgument2, ...);
*
* @class ns.widget.wearable.Page
* @extends ns.widget.core.Page
* @author hyunkook cho <hk0713.cho@samsung.com>
*/
(function (document, ns) {
"use strict";
/**
* Alias for {@link ns.widget.BaseWidget}
* @property {Object} BaseWidget
* @member ns.widget.core.Page
* @private
* @static
*/
var CorePage = ns.widget.core.Page,
/**
* Alias for {@link ns.util}
* @property {Object} util
* @member ns.widget.wearable.Page
* @private
* @static
*/
util = ns.util,
/**
* Alias for {@link ns.util.DOM}
* @property {Object} doms
* @member ns.widget.wearable.Page
* @private
* @static
*/
doms = util.DOM,
/**
* Alias for {@link ns.util.selectors}
* @property {Object} selectors
* @member ns.widget.wearable.Page
* @private
* @static
*/
selectors = util.selectors,
/**
* Alias for {@link ns.util.object}
* @property {Object} object
* @member ns.widget.wearable.Page
* @private
* @static
*/
object = ns.util.object,
/**
* Alias for {@link ns.event}
* @property {Object} object
* @member ns.widget.wearable.Page
* @private
* @static
*/
utilsEvents = ns.event,
/**
* Alias for {@link ns.event.gesture}
* @property {Object} object
* @member ns.widget.wearable.Page
* @private
* @static
*/
Gesture = utilsEvents.gesture,
/**
* Alias for {@link ns.engine}
* @property {Object} engine
* @member ns.widget.wearable.Page
* @private
* @static
*/
engine = ns.engine,
scrollBarType = {
CIRCLE: "tizen-circular-scrollbar"
},
Page = function () {
var self = this;
CorePage.call(self);
self._contentStyleAttributes = ["height", "width", "minHeight", "marginTop", "marginBottom"];
},
/**
* Dictionary for page related css class names
* @property {Object} classes
* @member ns.widget.core.Page
* @static
* @readonly
*/
classes = object.merge({
uiHeader: "ui-header",
uiContent: "ui-content",
uiPageScroll: "ui-scroll-on",
uiScroller: "ui-scroller",
uiFixed: "ui-fixed"
}, CorePage.classes),
prototype = new CorePage();
/**
* Configure Page Widget
* @method _configure
* @member ns.widget.wearable.Page
*/
prototype._configure = function () {
CorePage.prototype._configure.call(this);
this.options.enablePageScroll = ns.getConfig("enablePageScroll");
};
/**
* Sets top-bottom css attributes for content element
* to allow it to fill the page dynamically
* @method _contentFill
* @member ns.widget.wearable.Page
*/
prototype._contentFill = function () {
var self = this,
option = self.options,
element = self.element,
screenWidth = window.innerWidth,
screenHeight = window.innerHeight,
pageScrollSelector = classes.uiPageScroll,
children = [].slice.call(element.children),
elementStyle = element.style,
scroller,
content,
fragment,
firstChild;
elementStyle.width = screenWidth + "px";
elementStyle.height = screenHeight + "px";
if (option.enablePageScroll === true && !element.querySelector("." + classes.uiScroller)) {
element.classList.add(pageScrollSelector);
scroller = document.createElement("div");
scroller.classList.add(classes.uiScroller);
fragment = document.createDocumentFragment();
children.forEach( function(value) {
if ( selectors.matchesSelector(value, ".ui-header:not(.ui-fixed), .ui-content, .ui-footer:not(.ui-fixed)")) {
fragment.appendChild(value);
}
});
if (element.children.length > 0 && element.children[0].classList.contains(classes.uiHeader)) {
doms.insertNodeAfter(element.children[0], scroller);
} else {
element.insertBefore(scroller, element.firstChild);
}
firstChild = fragment.firstChild;
scroller.appendChild(fragment);
}
if (tau.support.shape.circle) {
if (scroller) {
scroller.setAttribute(scrollBarType.CIRCLE, "");
}
content = element.querySelector("." + classes.uiContent);
if (content) {
content.setAttribute(scrollBarType.CIRCLE, "");
}
}
};
prototype.getScroller = function() {
var element = this.element,
scroller = element.querySelector("." + classes.uiScroller);
return scroller || element.querySelector("." + classes.uiContent) || element;
};
prototype._destroy = function () {
CorePage.prototype._destroy.call(this);
};
Page.prototype = prototype;
// definition
ns.widget.wearable.Page = Page;
engine.defineWidget(
"Page",
"[data-role=page],.ui-page",
[
"layout",
"focus",
"blur",
"setActive",
"isActive"
],
Page,
"wearable",
true
);
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Callback Utility
* Class creates a callback list
*
* Create a callback list using the following parameters:
* options: an optional list of space-separated options that will change how
* the callback list behaves or a more traditional option object
*
* By default a callback list will act like an event callback list and can be
* "fired" multiple times.
*
* Possible options:
*
* once: will ensure the callback list can only be fired once (like a Deferred)
*
* memory: will keep track of previous values and will call any callback added
* after the list has been fired right away with the latest "memorized"
* values (like a Deferred)
*
* unique: will ensure a callback can only be added once (no duplicate in the list)
*
* stopOnFalse: interrupt callings when a callback returns false
* @class ns.util.callbacks
*/
(function (window, document, ns) {
"use strict";
ns.util.callbacks = function (orgOptions) {
var object = ns.util.object,
options = object.copy(orgOptions),
/**
* Alias to Array.slice function
* @method slice
* @member ns.util.callbacks
* @private
*/
slice = [].slice,
/**
* Last fire value (for non-forgettable lists)
* @property {Object} memory
* @member ns.util.callbacks
* @private
*/
memory,
/**
* Flag to know if list was already fired
* @property {boolean} fired
* @member ns.util.callbacks
* @private
*/
fired,
/**
* Flag to know if list is currently firing
* @property {boolean} firing
* @member ns.util.callbacks
* @private
*/
firing,
/**
* First callback to fire (used internally by add and fireWith)
* @property {number} [firingStart=0]
* @member ns.util.callbacks
* @private
*/
firingStart,
/**
* End of the loop when firing
* @property {number} firingLength
* @member ns.util.callbacks
* @private
*/
firingLength,
/**
* Index of currently firing callback (modified by remove if needed)
* @property {number} firingIndex
* @member ns.util.callbacks
* @private
*/
firingIndex,
/**
* Actual callback list
* @property {Array} list
* @member ns.util.callbacks
* @private
*/
list = [],
/**
* Stack of fire calls for repeatable lists
* @property {Array} stack
* @member ns.util.callbacks
* @private
*/
stack = !options.once && [],
fire,
add,
self = {
/**
* Add a callback or a collection of callbacks to the list
* @method add
* @param {..Function} list
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
add: function () {
if (list) {
// First, we save the current length
var start = list.length;
add(arguments);
// Do we need to add the callbacks to the
// current firing batch?
if (firing) {
firingLength = list.length;
// With memory, if we're not firing then
// we should call right away
} else if (memory) {
firingStart = start;
fire(memory);
}
}
return this;
},
/**
* Remove a callback from the list
* @method remove
* @param {..Function} list
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
remove: function () {
if (list) {
slice.call(arguments).forEach(function (arg) {
var index = list.indexOf(arg);
while (index > -1) {
list.splice(index, 1);
// Handle firing indexes
if (firing) {
if (index <= firingLength) {
firingLength--;
}
if (index <= firingIndex) {
firingIndex--;
}
}
index = list.indexOf(arg, index);
}
});
}
return this;
},
/**
* Check if a given callback is in the list.
* If no argument is given,
* return whether or not list has callbacks attached.
* @method has
* @param {Funciton} fn
* @return {boolean}
* @member ns.util.callbacks
*/
has: function (fn) {
return fn ? !!list && list.indexOf(fn) > -1 : !!(list && list.length);
},
/**
* Remove all callbacks from the list
* @method empty
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
empty: function () {
list = [];
firingLength = 0;
return this;
},
/**
* Have the list do nothing anymore
* @method disable
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
disable: function () {
list = stack = memory = undefined;
return this;
},
/**
* Is it disabled?
* @method disabled
* @return {boolean}
* @member ns.util.callbacks
*/
disabled: function () {
return !list;
},
/**
* Lock the list in its current state
* @method lock
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
lock: function () {
stack = undefined;
if (!memory) {
self.disable();
}
return this;
},
/**
* Is it locked?
* @method locked
* @return {boolean} stack
* @member ns.util.callbacks
*/
locked: function () {
return !stack;
},
/**
* Call all callbacks with the given context and
* arguments
* @method fireWith
* @param {Object} context
* @param {Array} args
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
fireWith: function (context, args) {
if (list && (!fired || stack)) {
args = args || [];
args = [context, args.slice ? args.slice() : args];
if (firing) {
stack.push(args);
} else {
fire(args);
}
}
return this;
},
/**
* Call all the callbacks with the given arguments
* @method fire
* @param {...*} argument
* @return {ns.util.callbacks} self
* @chainable
* @member ns.util.callbacks
*/
fire: function () {
self.fireWith(this, arguments);
return this;
},
/**
* To know if the callbacks have already been called at
* least once
* @method fired
* @return {booblean}
* @chainable
* @member ns.util.callbacks
*/
fired: function () {
return !!fired;
}
};
/**
* Adds functions to the callback list
* @method add
* @param {...*} argument
* @member ns.util.bezierCurve
* @private
*/
add = function (args) {
slice.call(args).forEach(function (arg) {
var type = typeof arg;
if (type === "function") {
if (!options.unique || !self.has(arg)) {
list.push(arg);
}
} else if (arg && arg.length && type !== "string") {
// Inspect recursively
add(arg);
}
});
};
/**
* Fire callbacks
* @method fire
* @param {Array} data
* @member ns.util.bezierCurve
* @private
*/
fire = function (data) {
memory = options.memory && data;
fired = true;
firingIndex = firingStart || 0;
firingStart = 0;
firingLength = list.length;
firing = true;
while (list && firingIndex < firingLength) {
if (list[firingIndex].apply(data[0], data[1]) === false && options.stopOnFalse) {
memory = false; // To prevent further calls using add
break;
}
firingIndex++;
}
firing = false;
if (list) {
if (stack) {
if (stack.length) {
fire(stack.shift());
}
} else if (memory) {
list = [];
} else {
self.disable();
}
}
};
return self;
};
}(window, window.document, ns));
/*global window, define, RegExp */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Deferred Utility
* Class creates object which can call registered callback depend from
* state of object..
* @class ns.util.deferred
* @author Tomasz Lukawski <t.lukawski@samsung.com>
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Piotr Karny <p.karny@samsung.com>
*/(function (window, document, ns) {
"use strict";
var Deferred = function (callback) {
var callbacks = ns.util.callbacks,
object = ns.util.object,
/**
* Register additional action for deferred object
* @property {Array} tuples
* @member ns.util.deferred
* @private
*/
tuples = [
// action, add listener, listener list, final state
["resolve", "done", callbacks({once: true, memory: true}), "resolved"],
["reject", "fail", callbacks({once: true, memory: true}), "rejected"],
["notify", "progress", callbacks({memory: true})]
],
state = "pending",
deferred = {},
promise = {
/**
* Determine the current state of a Deferred object.
* @method state
* @return {"pending" | "resolved" | "rejected"} representing the current state
* @member ns.util.deferred
*/
state: function () {
return state;
},
/**
* Add handlers to be called when the Deferred object
* is either resolved or rejected.
* @method always
* @param {...Function}
* @return {ns.util.deferred} self
* @member ns.util.deferred
*/
always: function () {
deferred.done(arguments).fail(arguments);
return this;
},
/**
* Add handlers to be called when the Deferred object
* is resolved, rejected, or still in progress.
* @method then
* @param {?Function} callback assign when done
* @param {?Function} callback assign when fail
* @param {?Function} callback assign when progress
* @return {Object} returns a new promise
* @member ns.util.deferred
*/
then: function () {/* fnDone, fnFail, fnProgress */
var functions = arguments;
return new Deferred(function (newDefer) {
tuples.forEach(function (tuple, i) {
var fn = (typeof functions[i] === 'function') && functions[i];
// deferred[ done | fail | progress ] for forwarding actions to newDefer
deferred[tuple[1]](function () {
var returned = fn && fn.apply(this, arguments);
if (returned && (typeof returned.promise === 'function')) {
returned.promise()
.done(newDefer.resolve)
.fail(newDefer.reject)
.progress(newDefer.notify);
} else {
newDefer[tuple[0] + "With"](this === promise ? newDefer.promise() : this, fn ? [returned] : arguments);
}
});
});
functions = null;
}).promise();
},
/**
* Get a promise for this deferred. If obj is provided,
* the promise aspect is added to the object
* @method promise
* @param {Object} obj
* @return {Object} return a Promise object
* @member ns.util.deferred
*/
promise: function (obj) {
if (obj) {
return object.merge(obj, promise);
}
return promise;
}
};
/**
* alias for promise.then, Keep pipe for back-compat
* @method pipe
* @member ns.util.deferred
*/
promise.pipe = promise.then;
// Add list-specific methods
tuples.forEach(function (tuple, i) {
var list = tuple[2],
stateString = tuple[3];
// promise[ done | fail | progress ] = list.add
promise[tuple[1]] = list.add;
// Handle state
if (stateString) {
list.add(function () {
// state = [ resolved | rejected ]
state = stateString;
// [ reject_list | resolve_list ].disable; progress_list.lock
}, tuples[i ^ 1][2].disable, tuples[2][2].lock);
}
// deferred[ resolve | reject | notify ]
deferred[tuple[0]] = function () {
deferred[tuple[0] + "With"](this === deferred ? promise : this, arguments);
return this;
};
deferred[tuple[0] + "With"] = list.fireWith;
});
// Make the deferred a promise
promise.promise(deferred);
// Call given func if any
if (callback) {
callback.call(deferred, deferred);
}
// All done!
return deferred;
};
ns.util.deferred = Deferred;
}(window, window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # Popup Widget
*
* @author Hyunkook Cho <hk0713.cho@samsung.com>
* @class ns.widget.core.Popup
* @extends ns.widget.Popup
*/
(function (ns) {
"use strict";
/**
* Alias for {@link ns.widget.BaseWidget}
* @property {Function} BaseWidget
* @member ns.widget.core.Popup
* @private
*/
var BaseWidget = ns.widget.BaseWidget,
/**
* Alias for class ns.engine
* @property {ns.engine} engine
* @member ns.widget.core.Popup
* @private
*/
engine = ns.engine,
/**
* Alias for class ns.util.object
* @property {Object} objectUtils
* @member ns.widget.core.Popup
* @private
*/
objectUtils = ns.util.object,
/**
* Alias for class ns.util.deferred
* @property {Object} UtilDeferred
* @member ns.widget.core.Popup
* @private
*/
UtilDeferred = ns.util.deferred,
/**
* Alias for class ns.util.selectors
* @property {Object} utilSelector
* @member ns.widget.core.Popup
* @private
*/
utilSelector = ns.util.selectors,
/**
* Alias for class ns.event
* @property {Object} eventUtils
* @member ns.widget.core.Popup
* @private
*/
eventUtils = ns.event,
Popup = function () {
var self = this,
ui = {};
self.selectors = selectors;
self.options = objectUtils.merge({}, Popup.defaults);
self.storedOptions = null;
/**
* Popup state flag
* @property {0|1|2|3} [state=null]
* @member ns.widget.core.Popup
* @private
*/
self.state = states.CLOSED;
ui.overlay = null;
ui.header = null;
ui.footer = null;
ui.content = null;
ui.container = null;
ui.wrapper = null;
self._ui = ui;
// event callbacks
self._callbacks = {};
},
/**
* Object with default options
* @property {Object} defaults
* @property {string} [options.transition="none"] Sets the default transition for the popup.
* @property {string} [options.positionTo="window"] Sets the element relative to which the popup will be centered.
* @property {boolean} [options.dismissible=true] Sets whether to close popup when a popup is open to support the back button.
* @property {boolean} [options.overlay=true] Sets whether to show overlay when a popup is open.
* @property {boolean|string} [options.header=false] Sets content of header.
* @property {boolean|string} [options.footer=false] Sets content of footer.
* @property {string} [options.content=null] Sets content of popup.
* @property {string} [options.overlayClass=""] Sets the custom class for the popup background, which covers the entire window.
* @property {string} [options.closeLinkSelector="a[data-rel='back']"] Sets selector for close buttons in popup.
* @property {boolean} [options.history=true] Sets whether to alter the url when a popup is open to support the back button.
* @member ns.widget.core.Popup
* @static
*/
defaults = {
transition: "none",
dismissible: true,
overlay: true,
header: false,
footer: false,
content: null,
overlayClass: "",
closeLinkSelector: "[data-rel='back']",
history: true
},
states = {
DURING_OPENING: 0,
OPENED: 1,
DURING_CLOSING: 2,
CLOSED: 3
},
CLASSES_PREFIX = "ui-popup",
/**
* Dictionary for popup related css class names
* @property {Object} classes
* @member ns.widget.core.Popup
* @static
*/
classes = {
popup: CLASSES_PREFIX,
active: CLASSES_PREFIX + "-active",
overlay: CLASSES_PREFIX + "-overlay",
header: CLASSES_PREFIX + "-header",
footer: CLASSES_PREFIX + "-footer",
content: CLASSES_PREFIX + "-content",
wrapper: CLASSES_PREFIX + "-wrapper",
build: "ui-build"
},
/**
* Dictionary for popup related selectors
* @property {Object} selectors
* @member ns.widget.core.Popup
* @static
*/
selectors = {
header: "." + classes.header,
content: "." + classes.content,
footer: "." + classes.footer
},
EVENTS_PREFIX = "popup",
/**
* Dictionary for popup related events
* @property {Object} events
* @member ns.widget.core.Popup
* @static
*/
events = {
/**
* Triggered when the popup has been created in the DOM (via ajax or other) but before all widgets have had an opportunity to enhance the contained markup.
* @event popupshow
* @member ns.widget.core.Popup
*/
show: EVENTS_PREFIX + "show",
/**
* Triggered on the popup after the transition animation has completed.
* @event popuphide
* @member ns.widget.core.Popup
*/
hide: EVENTS_PREFIX + "hide",
/**
* Triggered on the popup we are transitioning to, before the actual transition animation is kicked off.
* @event popupbeforeshow
* @member ns.widget.core.Popup
*/
before_show: EVENTS_PREFIX + "beforeshow",
/**
* Triggered on the popup we are transitioning away from, before the actual transition animation is kicked off.
* @event popupbeforehide
* @member ns.widget.core.Popup
*/
before_hide: EVENTS_PREFIX + "beforehide"
},
prototype = new BaseWidget();
Popup.classes = classes;
Popup.events = events;
Popup.defaults = defaults;
/**
* Build the content of popup
* @method _buildContent
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Popup
*/
prototype._buildContent = function (element) {
var self = this,
ui = self._ui,
selectors = self.selectors,
options = self.options,
content = ui.content || element.querySelector(selectors.content),
footer = ui.footer || element.querySelector(selectors.footer),
elementChildren = [].slice.call(element.childNodes),
elementChildrenLength = elementChildren.length,
i,
node;
if (!content) {
content = document.createElement("div");
content.className = classes.content;
for (i = 0; i < elementChildrenLength; ++i) {
node = elementChildren[i];
if (node !== ui.footer && node !== ui.header) {
content.appendChild(node);
}
}
if (typeof options.content === "string") {
content.innerHTML = options.content;
}
element.insertBefore(content, footer);
}
content.classList.add(classes.content);
ui.content = content;
};
/**
* Build the header of popup
* @method _buildHeader
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Popup
*/
prototype._buildHeader = function (element) {
var self = this,
ui = self._ui,
options = self.options,
selectors = self.selectors,
content = ui.content || element.querySelector(selectors.content),
header = ui.header || element.querySelector(selectors.header);
if (!header && options.header !== false) {
header = document.createElement("div");
header.className = classes.header;
if (typeof options.header !== "boolean") {
header.innerHTML = options.header;
}
element.insertBefore(header, content);
}
if (header) {
header.classList.add(classes.header);
}
ui.header = header;
};
/**
* Set the header of popup.
* This function is called by function "option" when the option "header" is set.
* @method _setHeader
* @param {HTMLElement} element
* @param {boolean|string} value
* @protected
* @member ns.widget.core.Popup
*/
prototype._setHeader = function (element, value) {
var self = this,
ui = self._ui,
header = ui.header;
if (header) {
header.parentNode.removeChild(header);
ui.header = null;
}
self.options.header = value;
self._buildHeader(ui.container);
};
/**
* Build the footer of popup
* @method _buildFooter
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Popup
*/
prototype._buildFooter = function (element) {
var self = this,
ui = self._ui,
options = self.options,
footer = ui.footer || element.querySelector(self.selectors.footer);
if (!footer && options.footer !== false) {
footer = document.createElement("div");
footer.className = classes.footer;
if (typeof options.footer !== "boolean") {
footer.innerHTML = options.footer;
}
element.appendChild(footer);
}
if (footer) {
footer.classList.add(classes.footer);
}
ui.footer = footer;
};
/**
* Set the footer of popup.
* This function is called by function "option" when the option "footer" is set.
* @method _setFooter
* @param {HTMLElement} element
* @param {boolean|string} value
* @protected
* @member ns.widget.core.Popup
*/
prototype._setFooter = function (element, value) {
var self = this,
ui = self._ui,
footer = ui.footer;
if (footer) {
footer.parentNode.removeChild(footer);
ui.footer = null;
}
self.options.footer = value;
self._buildFooter(ui.container);
};
/**
* Build structure of Popup widget
* @method _build
* @param {HTMLElement} element of popup
* @return {HTMLElement}
* @protected
* @member ns.widget.Popup
*/
prototype._build = function (element) {
var self = this,
ui = self._ui,
wrapper,
child = element.firstChild;
// set class for element
element.classList.add(classes.popup);
// create wrapper
wrapper = document.createElement("div");
wrapper.classList.add(classes.wrapper);
ui.wrapper = wrapper;
ui.container = wrapper;
// move all children to wrapper
while (child) {
wrapper.appendChild(child);
child = element.firstChild;
}
// add wrapper and arrow to popup element
element.appendChild(wrapper);
// build header, footer and content
this._buildHeader(ui.container);
this._buildFooter(ui.container);
this._buildContent(ui.container);
// set overlay
this._setOverlay(element, this.options.overlay);
return element;
};
/**
* Set overlay
* @method _setOverlay
* @param {HTMLElement} element
* @param {boolean} enable
* @protected
* @member ns.widget.Popup
*/
prototype._setOverlay = function(element, enable) {
var self = this,
overlayClass = self.options.overlayClass,
ui = self._ui,
overlay = ui.overlay;
// if this popup is not connected with slider,
// we create overlay, which is invisible when
// the value of option overlay is false
/// @TODO: get class from widget
if (!element.classList.contains("ui-slider-popup")) {
// create overlay
if (!overlay) {
overlay = document.createElement("div");
element.parentNode.insertBefore(overlay, element);
ui.overlay = overlay;
}
overlay.className = classes.overlay + (overlayClass ? " " + overlayClass : "");
if (enable) {
overlay.style.opacity = "";
} else {
// if option is set on "false", the overlay is not visible
overlay.style.opacity = 0;
}
}
};
/**
* Returns the state of the popup
* @method _isActive
* @protected
* @member ns.widget.core.Popup
*/
prototype._isActive = function () {
var state = this.state;
return state === states.DURING_OPENING || state === states.OPENED;
};
/**
* Returns true if popup is already opened and visible
* @method _isActive
* @protected
* @member ns.widget.core.Popup
*/
prototype._isOpened = function () {
return this.state === states.OPENED;
};
/**
* Init widget
* @method _init
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Popup
*/
prototype._init = function(element) {
var self = this,
selectors = self.selectors,
ui = self._ui;
ui.header = ui.header || element.querySelector(selectors.header);
ui.footer = ui.footer || element.querySelector(selectors.footer);
ui.content = ui.content || element.querySelector(selectors.content);
ui.wrapper = ui.wrapper || element.querySelector("." + classes.wrapper);
ui.container = ui.wrapper || element;
// @todo - use selector from page's definition in engine
ui.page = utilSelector.getClosestByClass(element, "ui-page") || window;
};
/**
* Set the state of the popup
* @method _setActive
* @param {boolean} active
* @protected
* @member ns.widget.core.Popup
*/
prototype._setActive = function (active) {
var self = this,
activeClass = classes.active,
elementClassList = self.element.classList,
route = engine.getRouter().getRoute("popup"),
options;
// NOTE: popup's options object is stored in window.history at the router module,
// and this window.history can't store DOM element object.
options = objectUtils.merge({}, self.options, {positionTo: null, link: null});
// set state of popup and add proper class
if (active) {
// set global variable
route.setActive(self, options);
// add proper class
elementClassList.add(activeClass);
// set state of popup 358
self.state = states.OPENED;
} else {
// no popup is opened, so set global variable on "null"
route.setActive(null, options);
// remove proper class
elementClassList.remove(activeClass);
// set state of popup
self.state = states.CLOSED;
}
};
/**
* Bind events
* @method _bindEvents
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Popup
*/
prototype._bindEvents = function () {
var self = this,
closeButtons = self.element.querySelectorAll(self.options.closeLinkSelector);
self._ui.page.addEventListener("pagebeforehide", self, false);
window.addEventListener("resize", self, false);
eventUtils.on(closeButtons, "click", self, false);
self._bindOverlayEvents();
};
/**
* Bind "click" event for overlay
* @method _bindOverlayEvents
* @protected
* @member ns.widget.core.Popup
*/
prototype._bindOverlayEvents = function () {
var overlay = this._ui.overlay;
if (overlay) {
overlay.addEventListener("click", this, false);
}
};
/**
* Unbind "click" event for overlay
* @method _bindOverlayEvents
* @protected
* @member ns.widget.core.Popup
*/
prototype._unbindOverlayEvents = function () {
var overlay = this._ui.overlay;
if (overlay) {
overlay.removeEventListener("click", this, false);
}
};
/**
* Unbind events
* @method _bindEvents
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.Popup
*/
prototype._unbindEvents = function () {
var self = this;
self._ui.page.removeEventListener("pagebeforehide", self, false);
window.removeEventListener("resize", self, false);
self._unbindOverlayEvents();
};
/**
* Layouting popup structure
* @method layout
* @member ns.widget.core.Popup
*/
prototype._layout = function (element) {
};
/**
* Open the popup
* @method open
* @param {Object=} [options]
* @param {string=} [options.transition] options.transition
* @member ns.widget.core.Popup
*/
prototype.open = function (options) {
var self = this,
newOptions;
if (!self._isActive()) {
/*
* Some passed options on open need to be kept until popup closing.
* For example, trasition parameter should be kept for closing animation.
* On the other hand, fromHashChange or x, y parameter should be removed.
* We store options and restore them on popup closing.
*/
self._storeOpenOptions(options);
newOptions = objectUtils.merge(self.options, options);
if (!newOptions.dismissible) {
engine.getRouter().lock();
}
self._show(newOptions);
}
};
/**
* Close the popup
* @method close
* @param {Object=} [options]
* @param {string=} [options.transition]
* @member ns.widget.core.Popup
*/
prototype.close = function (options) {
var self = this,
newOptions = objectUtils.merge(self.options, options);
if (self._isActive()) {
if (!newOptions.dismissible) {
engine.getRouter().unlock();
}
self._hide(newOptions);
}
};
/**
* Store Open options.
* @method _storeOpenOptions
* @param {object} options
* @protected
* @member ns.widget.core.Popup
*/
prototype._storeOpenOptions = function (options) {
var self = this,
oldOptions = self.options,
storedOptions = {},
key;
for (key in options) {
if (options.hasOwnProperty(key)) {
storedOptions[key] = oldOptions[key];
}
}
self.storedOptions = storedOptions;
};
/**
* Restore Open options and remove some unnecessary ones.
* @method _storeOpenOptions
* @protected
* @member ns.widget.core.Popup
*/
prototype._restoreOpenOptions = function () {
var self = this,
options = self.options,
propertiesToRemove = ["x", "y", "fromHashChange"];
// we restore opening values of all options
options = objectUtils.merge(options, self.storedOptions);
// and remove all values which should not be stored
objectUtils.removeProperties(options, propertiesToRemove);
};
/**
* Show popup.
* @method _show
* @param {object} options
* @protected
* @member ns.widget.core.Popup
*/
prototype._show = function (options) {
var self = this,
transitionOptions = objectUtils.merge({}, options),
overlay = self._ui.overlay,
deferred;
// layouting
self._layout(self.element);
// change state of popup
self.state = states.DURING_OPENING;
// set transiton
transitionOptions.ext = " in ";
self.trigger(events.before_show);
// show overlay
if (overlay) {
overlay.style.display = "block";
}
// start opening animation
self._transition(transitionOptions, self._onShow.bind(self));
};
/**
* Show popup
* @method _onShow
* @protected
* @member ns.widget.core.Popup
*/
prototype._onShow = function() {
var self = this;
self._setActive(true);
self.trigger(events.show);
};
/**
* Hide popup
* @method _hide
* @param {object} options
* @protected
* @member ns.widget.core.Popup
*/
prototype._hide = function (options) {
var self = this,
isOpened = self._isOpened(),
callbacks = self._callbacks;
// change state of popup
self.state = states.DURING_CLOSING;
self.trigger(events.before_hide);
if (isOpened) {
// popup is opened, so we start closing animation
options.ext = " out ";
self._transition(options, self._onHide.bind(self));
} else {
// popup is active, but not opened yet (DURING_OPENING), so
// we stop opening animation
if (callbacks.transitionDeferred) {
callbacks.transitionDeferred.reject();
}
if (callbacks.animationEnd) {
callbacks.animationEnd();
}
// and set popup as inactive
self._onHide();
}
};
/**
* Hide popup
* @method _onHide
* @protected
* @member ns.widget.core.Popup
*/
prototype._onHide = function() {
var self = this,
overlay = self._ui.overlay;
self._setActive(false);
if (overlay) {
overlay.style.display = "";
}
self._restoreOpenOptions();
self.trigger(events.hide);
};
/**
* Handle events
* @method handleEvent
* @param {Event} event
* @member ns.widget.core.Popup
*/
prototype.handleEvent = function(event) {
var self = this;
switch(event.type) {
case "pagebeforehide":
// we need close active popup if exists
engine.getRouter().close(null, {transition: "none", rel: "popup"});
break;
case "resize":
self._onResize(event);
break;
case "click":
if ( event.target === self._ui.overlay ) {
self._onClickOverlay(event);
}
break;
}
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.core.Popup
*/
prototype._refresh = function() {
var self = this;
self._unbindOverlayEvents();
self._setOverlay(self.element, self.options.overlay);
self._bindOverlayEvents();
};
/**
* Callback function fires after clicking on overlay.
* @method _onClickOverlay
* @param {Event} event
* @protected
* @member ns.widget.core.Popup
*/
prototype._onClickOverlay = function(event) {
var options = this.options;
event.preventDefault();
event.stopPropagation();
if (options.dismissible) {
engine.getRouter().close();
}
};
/**
* Callback function fires on resizing
* @method _onResize
* @protected
* @member ns.widget.core.Popup
*/
prototype._onResize = function() {
if (this._isOpened()) {
this._refresh();
}
};
function clearAnimation(self, transitionClass, deferred) {
var element = self.element,
elementClassList = element.classList,
overlay = self._ui.overlay,
animationEndCallback = self._callbacks.animationEnd;
// remove callbacks on animation events
element.removeEventListener("animationend", animationEndCallback, false);
element.removeEventListener("webkitAnimationEnd", animationEndCallback, false);
element.removeEventListener("mozAnimationEnd", animationEndCallback, false);
element.removeEventListener("oAnimationEnd", animationEndCallback, false);
element.removeEventListener("msAnimationEnd", animationEndCallback, false);
// clear classes
transitionClass.split(" ").forEach(function (currentClass) {
currentClass = currentClass.trim();
if (currentClass.length > 0) {
elementClassList.remove(currentClass);
if (overlay) {
overlay.classList.remove(currentClass);
}
}
});
if (deferred.state() === "pending") {
// we resolve only pending (not rejected) deferred
deferred.resolve();
}
}
function setTransitionDeferred(self, resolve) {
var deferred = new UtilDeferred();
deferred.then(function() {
if (deferred === self._callbacks.transitionDeferred) {
resolve();
}
});
self._callbacks.transitionDeferred = deferred;
return deferred;
}
/**
* Animate popup opening/closing
* @method _transition
* @protected
* @param {Object} [options]
* @param {string=} [options.transition]
* @param {string=} [options.ext]
* @param {?Function} [resolve]
* @member ns.widget.core.Popup
*/
prototype._transition = function (options, resolve) {
var self = this,
transition = options.transition || self.options.transition || "none",
transitionClass = transition + options.ext,
element = self.element,
elementClassList = element.classList,
overlayClassList = self._ui.overlay.classList,
deferred,
animationEndCallback;
deferred = setTransitionDeferred(self, resolve);
if (transition !== "none") {
// set animationEnd callback
animationEndCallback = clearAnimation.bind(null, self, transitionClass, deferred);
self._callbacks.animationEnd = animationEndCallback;
// add animation callbacks
element.addEventListener("animationend", animationEndCallback, false);
element.addEventListener("webkitAnimationEnd", animationEndCallback, false);
element.addEventListener("mozAnimationEnd", animationEndCallback, false);
element.addEventListener("oAnimationEnd", animationEndCallback, false);
element.addEventListener("msAnimationEnd", animationEndCallback, false);
// add transition classes
transitionClass.split(" ").forEach(function (currentClass) {
currentClass = currentClass.trim();
if (currentClass.length > 0) {
elementClassList.add(currentClass);
overlayClassList.add(currentClass);
}
});
} else {
window.setTimeout(deferred.resolve, 0);
}
return deferred;
};
/**
* Destroy popup
* @method _destroy
* @protected
* @member ns.widget.core.Popup
*/
prototype._destroy = function() {
var self = this,
element = self.element,
ui = self._ui,
wrapper = ui.wrapper,
child;
if (wrapper) {
// restore all children from wrapper
child = wrapper.firstChild;
while (child) {
element.appendChild(child);
child = wrapper.firstChild;
}
if (wrapper.parentNode) {
wrapper.parentNode.removeChild(wrapper);
}
}
self._unbindEvents(element);
self._setOverlay(element, false);
ui.wrapper = null;
};
Popup.prototype = prototype;
ns.widget.core.Popup = Popup;
engine.defineWidget(
"Popup",
"[data-role='popup'], .ui-popup",
[
"open",
"close",
"reposition"
],
Popup,
"core"
);
}(ns));
/*global window, define */
/*
* Copyright (c) 2013 - 2014 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # Popup Widget
* Shows a pop-up window.
*
* The popup widget shows in the middle of the screen a list of items in a pop-up window. It automatically optimizes the pop-up window size within the screen. The following table describes the supported popup classes.
*
* ## Default selectors
* All elements with class *ui-popup* will be become popup widgets.
*
* The pop-up window can contain a header, content, and footer area like the page element.
*
* To open a pop-up window from a link, use the data-rel attribute in HTML markup as in the following code:
*
* @example
* <a href="#popup" class="ui-btn" data-rel="popup">Open popup when clicking this element.</a>
*
* The following table shows examples of various types of popups.
*
* The popup contains header, content and footer area
*
* ###HTML Examples
*
* #### Basic popup with header, content, footer
*
* @example
* <div class="ui-page">
* <div class="ui-popup">
* <div class="ui-popup-header">Power saving mode</div>
* <div class="ui-popup-content">
* Turning on Power
* saving mode will
* limit the maximum
* per
* </div>
* <div class="ui-popup-footer">
* <button id="cancel" class="ui-btn">Cancel</button>
* </div>
* </div>
* </div>
*
* #### Popup with 2 buttons in the footer
*
* @example
* <div id="2btnPopup" class="ui-popup">
* <div class="ui-popup-header">Delete</div>
* <div class="ui-popup-content">
* Delete the image?
* </div>
* <div class="ui-popup-footer ui-grid-col-2">
* <button id="2btnPopup-cancel" class="ui-btn">Cancel</button>
* <button id="2btnPopup-ok" class="ui-btn">OK</button>
* </div>
* </div>
*
* #### Popup with checkbox/radio
*
* If you want make popup with list checkbox(or radio) just include checkbox (radio) to popup and add class *ui-popup-checkbox-label* to popup element.
*
* @example
* <div id="listBoxPopup" class="ui-popup">
* <div class="ui-popup-header">When?</div>
* <div class="ui-popup-content" style="height:243px; overflow-y:scroll">
* <ul class="ui-listview">
* <li>
* <label for="check-1" class="ui-popup-checkbox-label">Yesterday</label>
* <input type="checkbox" name="checkset" id="check-1" />
* </li>
* <li>
* <label for="check-2" class="ui-popup-checkbox-label">Today</label>
* <input type="checkbox" name="checkset" id="check-2" />
* </li>
* <li>
* <label for="check-3" class="ui-popup-checkbox-label">Tomorrow</label>
* <input type="checkbox" name="checkset" id="check-3" />
* </li>
* </ul>
* <ul class="ui-listview">
* <li>
* <label for="radio-1" class="ui-popup-radio-label">Mandatory</label>
* <input type="radio" name="radioset" id="radio-1" />
* </li>
* <li>
* <label for="radio-2" class="ui-popup-radio-label">Optional</label>
* <input type="radio" name="radioset" id="radio-2" />
* </li>
* </ul>
* </div>
* <div class="ui-popup-footer">
* <button id="listBoxPopup-close" class="ui-btn">Close</button>
* </div>
* </div>
* </div>
*
* #### Popup with no header and footer
*
* @example
* <div id="listNoTitleNoBtnPopup" class="ui-popup">
* <div class="ui-popup-content" style="height:294px; overflow-y:scroll">
* <ul class="ui-listview">
* <li><a href="">Ringtones 1</a></li>
* <li><a href="">Ringtones 2</a></li>
* <li><a href="">Ringtones 3</a></li>
* </ul>
* </div>
* </div>
*
* #### Toast popup
*
* @example
* <div id="PopupToast" class="ui-popup ui-popup-toast">
* <div class="ui-popup-content">Saving contacts to sim on Samsung</div>
* </div>
*
* ### Create Option popup
*
* Popup inherits value of option positionTo from property data-position-to set in link.
*
* @example
* <!--definition of link, which opens popup and sets its position-->
* <a href="#popupOptionText" data-rel="popup" data-position-to="origin">Text</a>
* <!--definition of popup, which inherites property position from link-->
* <div id="popupOptionText" class="ui-popup">
* <div class="ui-popup-content">
* <ul class="ui-listview">
* <li><a href="#">Option 1</a></li>
* <li><a href="#">Option 2</a></li>
* <li><a href="#">Option 3</a></li>
* <li><a href="#">Option 4</a></li>
* </ul>
* </div>
* </div>
*
* ### Opening and closing popup
*
* To open popup from "a" link using html markup, use the following code:
*
* @example
* <div class="ui-page">
* <header class="ui-header">
* <h2 class="ui-title">Call menu</h2>
* </header>
* <div class="ui-content">
* <a href="#popup" class="ui-btn" data-rel="popup" >Open Popup</a>
* </div>
*
* <div id="popup" class="ui-popup">
* <div class="ui-popup-header">Power saving mode</div>
* <div class="ui-popup-content">
* Turning on Power
* saving mode will
* limit the maximum
* per
* </div>
* <div class="ui-popup-footer">
* <button id="cancel" class="ui-btn">Cancel</button>
* </div>
* </div>
*
* To open the popup widget from JavaScript use method *tau.openPopup(to)*
*
* @example
* tau.openPopup("popup")
*
* To close the popup widget from JavaScript use method *tau.openPopup(to)*
*
* @example
* tau.closePopup("popup")
*
* To find the currently active popup, use the ui-popup-active class.
*
* To bind the popup to a button, use the following code:
*
* @example
* <!--HTML code-->
* <div id="1btnPopup" class="ui-popup">
* <div class="ui-popup-header">Power saving mode</div>
* <div class="ui-popup-content">
* </div>
* <div class="ui-popup-footer">
* <button id="1btnPopup-cancel" class="ui-btn">Cancel</button>
* </div>
* </div>
* <script>
* // Popup opens with button click
* var button = document.getElementById("button");
* button.addEventListener("click", function() {
* tau.openPopup("#1btnPopup");
* });
*
* // Popup closes with Cancel button click
* document.getElementById("1btnPopup-cancel").addEventListener("click", function() {
* tau.closePopup();
* });
* </script>
*
* ## Manual constructor
* For manual creation of popup widget you can use constructor of widget from **tau** namespace:
*
* @example
* var popupElement = document.getElementById("popup"),
* popup = tau.widget.popup(buttonElement);
*
* Constructor has one require parameter **element** which are base **HTMLElement** to create widget. We recommend get this element by method *document.getElementById*.
*
* ## Options for Popup Widget
*
* Options for widget can be defined as _data-..._ attributes or give as parameter in constructor.
*
* You can change option for widget using method **option**.
*
* ## Methods
*
* To call method on widget you can use tau API:
*
* @example
* var popupElement = document.getElementById("popup"),
* popup = tau.widget.popup(buttonElement);
*
* popup.methodName(methodArgument1, methodArgument2, ...);
*
* ## Transitions
*
* By default, the framework doesn't apply transition. To set a custom transition effect, add the data-transition attribute to the link.
*
* @example
* <a href="index.html" data-rel="popup" data-transition="slideup">I'll slide up</a>
*
* Global configuration:
*
* @example
* gear.ui.defaults.popupTransition = "slideup";
*
* ### Transitions list
*
* - **none** Default value, no transition.
* - **slideup** Makes the content of the pop-up slide up.
*
* ## Handling Popup Events
*
* To use popup events, use the following code:
*
* @example
* <!--Popup html code-->
* <div id="popup" class="ui-popup">
* <div class="ui-popup-header"></div>
* <div class="ui-popup-content"></div>
* </div>
* </div>
* <script>
* // Use popup events
* var popup = document.getElementById("popup");
* popup.addEventListener("popupbeforecreate", function() {
* // Implement code for popupbeforecreate event
* });
* </script>
*
* Full list of available events is in [events list section](#events-list).
*
* @author Hyunkook Cho <hk0713.cho@samsung.com>
* @class ns.widget.core.Popup
* @extends ns.widget.core.BasePopup
*/
(function (window, document, ns) {
"use strict";
var Popup = ns.widget.core.Popup,
PopupPrototype = Popup.prototype,
engine = ns.engine,
objectUtils = ns.util.object,
domUtils = ns.util.DOM,
/**
* Object with default options
* @property {Object} defaults
* @property {string} [options.transition="none"] Sets the default transition for the popup.
* @property {string} [options.positionTo="window"] Sets the element relative to which the popup will be centered.
* @property {boolean} [options.dismissible=true] Sets whether to close popup when a popup is open to support the back button.
* @property {boolean} [options.overlay=true] Sets whether to show overlay when a popup is open.
* @property {string} [overlayClass=""] Sets the custom class for the popup background, which covers the entire window.
* @property {boolean} [options.history=true] Sets whether to alter the url when a popup is open to support the back button.
* @property {string} [options.arrow="l,t,r,b"] Sets directions of popup's arrow by priority ("l" for left, "t" for top,
* "r" for right, and "b" for bottom). The first one has the highest priority, the last one - the lowest. If you set arrow="t",
* then arrow will be placed at the top of popup container and the whole popup will be placed under cliced element.
* @property {string} [options.positionTo="window"] Sets the element relative to which the popup will be centered.
* @property {number} [options.distance=0] Sets the extra distance in px from clicked element.
* @property {HTMLElement|string} [options.link=null] Set the element or its id, under which popup should be placed.
* It only works with option positionTo="origin".
* @member ns.widget.core.ContextPopup
* @static
* @private
*/
defaults = {
arrow: "l,b,r,t",
positionTo: "window",
positionOriginCenter: false,
distance: 0,
link: null
},
ContextPopup = function () {
var self = this,
ui;
Popup.call(self);
// set options
self.options = objectUtils.merge(self.options, defaults);
// set ui
ui = self._ui || {};
ui.arrow = null;
self._ui = ui;
},
/**
* @property {Object} classes Dictionary for popup related css class names
* @member ns.widget.core.Popup
* @static
*/
CLASSES_PREFIX = "ui-popup",
classes = objectUtils.merge({}, Popup.classes, {
context: "ui-ctxpopup",
contextOverlay: "ui-ctxpopup-overlay",
arrow: "ui-arrow",
arrowDir: CLASSES_PREFIX + "-arrow-"
}),
/**
* @property {Object} events Dictionary for popup related events
* @member ns.widget.core.Popup
* @static
*/
events = objectUtils.merge({}, Popup.events, {
before_position: "beforeposition"
}),
positionTypes = {
WINDOW: "window",
ORIGIN: "origin",
ABSOLUTE: "absolute"
},
prototype = new Popup();
ContextPopup.defaults = objectUtils.merge({}, Popup.defaults, defaults);
ContextPopup.classes = classes;
ContextPopup.events = events;
ContextPopup.positionTypes = positionTypes;
/**
* Build structure of Popup widget
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.core.Popup
*/
prototype._build = function (element) {
var self = this,
ui = self._ui,
arrow;
// build elements of popup
PopupPrototype._build.call(self, element);
// set class for element
element.classList.add(classes.popup);
// create arrow
arrow = document.createElement("div");
arrow.appendChild(document.createElement("span"));
arrow.classList.add(classes.arrow);
ui.arrow = arrow;
// add arrow to popup element
element.appendChild(arrow);
return element;
};
/**
* Init widget
* @method _init
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._init = function(element) {
var self = this,
ui = self._ui;
PopupPrototype._init.call(this, element);
ui.arrow = ui.arrow || element.querySelector("." + classes.arrow);
};
/**
* Layouting popup structure
* @method layout
* @member ns.widget.core.ContextPopup
*/
prototype._layout = function (element) {
var self = this;
this._reposition();
PopupPrototype._layout.call(self, element);
};
/**
* Set positon and size of popup.
* @method _reposition
* @param {object} options
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._reposition = function(options) {
var self = this,
element = self.element,
ui = self._ui,
elementClassList = element.classList;
options = objectUtils.merge({}, self.options, options);
self.trigger(events.before_position, null, false);
elementClassList.add(classes.build);
// set height of content
self._setContentHeight();
// set class for contextpopup
if ((options.positionTo === "origin") && ui.overlay) {
ui.overlay.classList.add(classes.contextOverlay);
}
// set position of popup
self._placementCoords(options);
elementClassList.remove(classes.build);
};
/**
* Find the best positon of context popup.
* @method findBestPosition
* @param {ns.widget.core.ContextPopup} self
* @param {HTMLElement} clickedElement
* @private
* @member ns.widget.core.ContextPopup
*/
function findBestPosition(self, clickedElement) {
var options = self.options,
arrowsPriority = options.arrow.split(","),
element = self.element,
windowWidth = window.innerWidth,
windowHeight = window.innerHeight,
popupWidth = domUtils.getElementWidth(element, "outer"),
popupHeight = domUtils.getElementHeight(element, "outer"),
// offset coordinates of clicked element
clickElementRect = clickedElement.getBoundingClientRect(),
clickElementOffsetX = clickElementRect.left,
clickElementOffsetY = clickElementRect.top,
// width of visible part of clicked element
clickElementOffsetWidth = Math.min(clickElementRect.width,
windowWidth - clickElementOffsetX),
// height of visible part of clicked element
clickElementOffsetHeight = Math.min(clickElementRect.height,
windowHeight - clickElementOffsetY),
// params for all types of popup
// "l" - popup with arrow on the left side, "r" - right, "b" - bottom, "t" - top
// dir - this letter is added as a suffix of class to popup's element
// fixedPositionField - specifies which coordinate is changed for this type of popup
// fixedPositionFactor - factor, which specifies if size should be added or subtracted
// size - available size, which is needed for this type of popup (width or height)
// max - maximum size of available place
params = {
"l": {dir: "l", fixedPositionField: "x", fixedPositionFactor: 1,
size: popupWidth, max: clickElementOffsetX},
"r": {dir: "r", fixedPositionField: "x", fixedPositionFactor: -1,
size: popupWidth, max: windowWidth - clickElementOffsetX - clickElementOffsetWidth},
"b": {dir: "b", fixedPositionField: "y", fixedPositionFactor: -1,
size: popupHeight, max: clickElementOffsetY},
"t": {dir: "t", fixedPositionField: "y", fixedPositionFactor: 1,
size: popupHeight, max: windowHeight - clickElementOffsetY - clickElementOffsetHeight}
},
bestDirection,
direction,
bestOffsetInfo;
// set value of bestDirection on the first possible type or top
bestDirection = params[arrowsPriority[0]] || params.t;
arrowsPriority.forEach(function(key){
var param = params[key],
paramMax = param.max;
if (!direction) {
if (param.size < paramMax) {
direction = param;
} else if (paramMax > bestDirection.max) {
bestDirection = param;
}
}
});
if (!direction) {
direction = bestDirection;
if (direction.fixedPositionField === "x") {
popupWidth = direction.max;
} else {
popupHeight = direction.max;
}
}
// info about the best position without taking into account type of popup
bestOffsetInfo = {
x: clickElementOffsetX + clickElementOffsetWidth / 2 - popupWidth / 2,
y: clickElementOffsetY + clickElementOffsetHeight / 2 - popupHeight / 2,
w: popupWidth,
h: popupHeight,
dir: direction.dir
};
// check type of popup and correct value for "fixedPositionField" coordinate
bestOffsetInfo[direction.fixedPositionField] +=
(direction.fixedPositionField === "x" ?
(popupWidth + clickElementOffsetWidth) * direction.fixedPositionFactor :
(popupHeight + clickElementOffsetHeight) * direction.fixedPositionFactor)
/ 2 + options.distance * direction.fixedPositionFactor;
// fix min/max position
bestOffsetInfo.x = bestOffsetInfo.x < 0 ? 0 : bestOffsetInfo.x + bestOffsetInfo.w > windowWidth ? windowWidth - bestOffsetInfo.w : bestOffsetInfo.x;
bestOffsetInfo.y = bestOffsetInfo.y < 0 ? 0 : bestOffsetInfo.y + bestOffsetInfo.h > windowHeight ? windowHeight - bestOffsetInfo.h : bestOffsetInfo.y;
return bestOffsetInfo;
}
/**
* Find the best positon of arrow.
* @method adjustedPositionAndPlacementArrow
* @param {ns.widget.core.ContextPopup} self
* @param {Object} bestRectangle
* @param {number} x
* @param {number} y
* @private
* @member ns.widget.core.ContextPopup
*/
function adjustedPositionAndPlacementArrow(self, bestRectangle, x, y) {
var ui = self._ui,
wrapper = ui.wrapper,
arrow = ui.arrow,
popupElement = self.element,
arrowStyle = arrow.style,
windowWidth = window.innerWidth,
windowHeight = window.innerHeight,
wrapperRect = wrapper.getBoundingClientRect(),
arrowHalfWidth = arrow.offsetWidth / 2,
popupProperties = {
"padding-top": 0,
"padding-bottom": 0,
"padding-left": 0,
"padding-right": 0,
"border-top-width": 0,
"border-left-width": 0,
"box-sizing": null
},
wrapperProperties = {
"margin-top": 0,
"margin-bottom": 0,
"margin-left": 0,
"margin-right": 0,
"padding-top": 0,
"padding-bottom": 0,
"padding-left": 0,
"padding-right": 0
},
margins,
params = {
"t": {pos: x, min: "left", max: "right", posField: "x", valField: "w", styleField: "left"},
"b": {pos: x, min: "left", max: "right", posField: "x", valField: "w", styleField: "left"},
"l": {pos: y, min: "top", max: "bottom", posField: "y", valField: "h", styleField: "top"},
"r": {pos: y, min: "top", max: "bottom", posField: "y", valField: "h", styleField: "top"}
},
param = params[bestRectangle.dir],
surplus,
addPadding;
domUtils.extractCSSProperties(popupElement, popupProperties);
domUtils.extractCSSProperties(wrapper, wrapperProperties);
addPadding = popupProperties["box-sizing"] === "border-box";
margins = {
"t": popupProperties["padding-top"] + wrapperProperties["margin-top"] + wrapperProperties["padding-top"],
"b": popupProperties["padding-bottom"] + wrapperProperties["margin-bottom"] + wrapperProperties["padding-bottom"],
"l": popupProperties["padding-left"] + wrapperProperties["margin-left"] + wrapperProperties["padding-left"],
"r": popupProperties["padding-right"] + wrapperProperties["margin-right"] + wrapperProperties["padding-right"]
};
// value of coordinates of proper edge of wrapper
wrapperRect = {
// x-coordinate of left edge
left: margins.l + bestRectangle.x,
// x-coordinate of right edge
right: margins.l + wrapperRect.width + bestRectangle.x,
// y-coordinate of top edge
top: margins.t + bestRectangle.y,
// y-coordinate of bottom edge
bottom: wrapperRect.height + margins.t + bestRectangle.y
};
if (wrapperRect[param.min] > param.pos - arrowHalfWidth) {
surplus = bestRectangle[param.posField];
if (surplus > 0) {
bestRectangle[param.posField] = Math.max(param.pos - arrowHalfWidth, 0);
param.pos = bestRectangle[param.posField] + arrowHalfWidth;
} else {
param.pos = wrapperRect[param.min] + arrowHalfWidth;
}
} else if (wrapperRect[param.max] < param.pos + arrowHalfWidth) {
surplus = (param.valField === "w" ? windowWidth : windowHeight)
- (bestRectangle[param.posField] + bestRectangle[param.valField]);
if (surplus > 0) {
bestRectangle[param.posField] += Math.min(surplus, (param.pos + arrowHalfWidth) - wrapperRect[param.max]);
param.pos = bestRectangle[param.posField] + bestRectangle[param.valField] - arrowHalfWidth;
} else {
param.pos = wrapperRect[param.max] - arrowHalfWidth;
}
}
arrowStyle[param.styleField] = (param.pos - arrowHalfWidth - bestRectangle[param.posField] - (addPadding ? popupProperties["border-" + param.styleField + "-width"] : 0)) + "px";
return bestRectangle;
}
/**
* Set top, left and margin for popup's container.
* @method _placementCoordsWindow
* @param {HTMLElement} element
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._placementCoordsWindow = function(element) {
var elementStyle = element.style,
elementWidth = element.offsetWidth,
elementHeight = element.offsetHeight,
elementMarginTop = domUtils.getCSSProperty(element, "margin-top", 0, "float"),
elementTop = window.innerHeight - elementHeight - elementMarginTop;
elementStyle.top = elementTop + "px";
elementStyle.left = "50%";
elementStyle.marginLeft = -(elementWidth / 2) + "px";
};
/**
* Set top, left and margin for popup's container.
* @method _placementCoordsAbsolute
* @param {HTMLElement} element
* @param {number} x
* @param {number} y
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._placementCoordsAbsolute = function(element, x, y) {
var elementStyle = element.style,
elementWidth = element.offsetWidth,
elementHeight = element.offsetHeight;
elementStyle.top = y + "px";
elementStyle.left = x + "px";
elementStyle.marginTop = -(elementHeight / 2) + "px";
elementStyle.marginLeft = -(elementWidth / 2) + "px";
};
/**
* Find clicked element.
* @method _findClickedElement
* @param {number} x
* @param {number} y
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._findClickedElement = function(x, y) {
return document.elementFromPoint(x, y);
};
/**
* Emulate position of event for clicked element.
* @method emulatePositionOfClick
* @param {string} bestDirection direction of arrow
* @param {HTMLElement} clickedElement
* @private
* @member ns.widget.core.ContextPopup
*/
function emulatePositionOfClick(bestDirection, clickedElement) {
var clickedElementRect = clickedElement.getBoundingClientRect(),
position = {};
switch(bestDirection) {
case "l":
// the arrow will be on the left edge of container, so x-coordinate
// should have value equals to the position of right edge of clicked element
position.x = clickedElementRect.right;
// y-coordinate should have value equals to the position of top edge of clicked
// element plus half of its height
position.y = clickedElementRect.top + clickedElementRect.height / 2;
break;
case "r":
// the arrow will be on the right edge of container
position.x = clickedElementRect.left;
position.y = clickedElementRect.top + clickedElementRect.height / 2;
break;
case "t":
// the arrow will be on the top edge of container
position.x = clickedElementRect.left + clickedElementRect.width / 2;
position.y = clickedElementRect.bottom;
break;
case "b":
// the arrow will be on the bottom edge of container
position.x = clickedElementRect.left + clickedElementRect.width / 2;
position.y = clickedElementRect.top;
break;
}
return position;
}
prototype._placementCoordsOrigin = function (clickedElement, options) {
var self = this,
element = self.element,
elementStyle = element.style,
elementClassList = element.classList,
x = options.x,
y = options.y,
bestRectangle,
emulatedPosition,
arrowType,
elementHeight;
elementClassList.add(classes.context);
elementHeight = element.offsetHeight;
bestRectangle = findBestPosition(self, clickedElement);
arrowType = bestRectangle.dir;
elementClassList.add(classes.arrowDir + arrowType);
self._ui.arrow.setAttribute("type", arrowType);
if ((typeof x !== "number" && typeof y !== "number") || self.options.positionOriginCenter) {
// if we found element, which was clicked, but the coordinates of event
// was not available, we have to count these coordinates to the center of proper edge of element.
emulatedPosition = emulatePositionOfClick(arrowType, clickedElement);
x = emulatedPosition.x;
y = emulatedPosition.y;
}
bestRectangle = adjustedPositionAndPlacementArrow(self, bestRectangle, x, y);
if (elementHeight > bestRectangle.h) {
self._setContentHeight(bestRectangle.h);
}
elementStyle.left = bestRectangle.x + "px";
elementStyle.top = bestRectangle.y + "px";
};
prototype._placementCoordsElement = function (clickedElement, options) {
var self = this,
element = self.element,
elementStyle = element.style,
bestRectangle,
elementHeight;
element.classList.add(classes.context);
elementHeight = element.offsetHeight;
bestRectangle = findBestPosition(self, clickedElement);
if (elementHeight > bestRectangle.h) {
self._setContentHeight(bestRectangle.h);
}
elementStyle.left = bestRectangle.x + "px";
elementStyle.top = bestRectangle.y + "px";
};
/**
* Find and set the best position for popup.
* @method _placementCoords
* @param {object} options
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._placementCoords = function(options) {
var self = this,
positionTo = options.positionTo,
x = options.x,
y = options.y,
element = self.element,
elementHeight,
clickedElement,
link;
switch (positionTo) {
case positionTypes.ORIGIN:
// if we know x-coord and y-coord, we open the popup with arrow
link = options.link;
if (link) {
if (typeof link === "string") {
clickedElement = document.getElementById(link);
} else if (typeof link === "object") {
clickedElement = link;
}
} else if (typeof x === "number" && typeof y === "number") {
clickedElement = self._findClickedElement(x, y);
}
if (clickedElement) {
self._placementCoordsOrigin(clickedElement, options);
return;
}
break;
case positionTypes.WINDOW:
self._placementCoordsWindow(element);
return;
break;
case positionTypes.ABSOLUTE:
if (typeof x === "number" && typeof y === "number") {
self._placementCoordsAbsolute(element, x, y);
return;
}
break;
default:
// there is posible, that element or its id was given
if (typeof positionTo === "string") {
try {
clickedElement = document.querySelector(options.positionTo);
} catch(e) {}
} else if (typeof positionTo === "object") {
clickedElement = positionTo;
}
if (clickedElement) {
self._placementCoordsElement(clickedElement, options);
return;
}
break;
}
// if there was problem with setting position of popup, we set its position to window
self._placementCoordsWindow(element);
};
/**
* Set height for popup's container.
* @method _setContentHeight
* @param {number} maxHeight
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._setContentHeight = function(maxHeight) {
var self = this,
element = self.element,
content = self._ui.content,
contentStyle,
contentHeight,
elementOffsetHeight;
if (content) {
contentStyle = content.style;
if (contentStyle.height || contentStyle.minHeight) {
contentStyle.height = "";
contentStyle.minHeight = "";
}
maxHeight = maxHeight || window.innerHeight;
contentHeight = content.offsetHeight;
elementOffsetHeight = element.offsetHeight;
if (elementOffsetHeight > maxHeight) {
contentHeight -= (elementOffsetHeight - maxHeight);
contentStyle.height = contentHeight + "px";
contentStyle.minHeight = contentHeight + "px";
}
}
};
/**
* Hide popup.
* @method _onHide
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._onHide = function() {
var self = this,
ui = self._ui,
element = self.element,
elementClassList = element.classList,
content = ui.content,
arrow = ui.arrow;
elementClassList.remove(classes.context);
["l", "r", "b", "t"].forEach(function(key) {
elementClassList.remove(classes.arrowDir + key);
});
// we remove styles for element, which are changed
// styles for container, header and footer are left unchanged
element.removeAttribute("style");
arrow.removeAttribute("style");
PopupPrototype._onHide.call(self);
};
/**
* Destroy popup.
* @method _destroy
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._destroy = function() {
var self = this,
element = self.element,
ui = self._ui,
arrow = ui.arrow;
PopupPrototype._destroy.call(self);
if (arrow && arrow.parentNode) {
arrow.parentNode.removeChild(arrow);
}
ui.arrow = null;
};
/**
* Set new position for popup.
* @method reposition
* @param options
* @param options.x
* @param options.y
* @param options.positionTo
* @member ns.widget.core.ContextPopup
*/
prototype.reposition = function(options) {
if (this._isActive()) {
this._reposition(options);
}
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.core.ContextPopup
*/
prototype._refresh = function() {
if (this._isActive()) {
PopupPrototype._refresh.call(this);
this.reposition(this.options);
}
};
ContextPopup.prototype = prototype;
ns.widget.core.ContextPopup = ContextPopup;
engine.defineWidget(
"Popup",
"[data-role='popup'], .ui-popup",
[
"open",
"close",
"reposition"
],
ContextPopup,
"core",
true
);
// @remove
// THIS IS ONLY FOR COMPATIBILITY
ns.widget.popup = ns.widget.Popup;
}(window, window.document, ns));
/*global window, define */
/*
* Copyright (c) 2013 - 2014 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* # Popup Widget
* Shows a pop-up window.
*
* The popup widget shows in the middle of the screen a list of items in a pop-up window. It automatically optimizes the pop-up window size within the screen. The following table describes the supported popup classes.
*
* ## Default selectors
* All elements with class *ui-popup* will be become popup widgets.
*
* The pop-up window can contain a header, content, and footer area like the page element.
*
* To open a pop-up window from a link, use the data-rel attribute in HTML markup as in the following code:
*
* @example
* <a href="#popup" class="ui-btn" data-rel="popup">Open popup when clicking this element.</a>
*
* The following table shows examples of various types of popups.
*
* The popup contains header, content and footer area
*
* ###HTML Examples
*
* #### Basic popup with header, content, footer
*
* @example
* <div class="ui-page">
* <div class="ui-popup">
* <div class="ui-popup-header">Power saving mode</div>
* <div class="ui-popup-content">
* Turning on Power
* saving mode will
* limit the maximum
* per
* </div>
* <div class="ui-popup-footer">
* <button id="cancel" class="ui-btn">Cancel</button>
* </div>
* </div>
* </div>
*
* #### Popup with 2 buttons in the footer
*
* @example
* <div id="2btnPopup" class="ui-popup">
* <div class="ui-popup-header">Delete</div>
* <div class="ui-popup-content">
* Delete the image?
* </div>
* <div class="ui-popup-footer ui-grid-col-2">
* <button id="2btnPopup-cancel" class="ui-btn">Cancel</button>
* <button id="2btnPopup-ok" class="ui-btn">OK</button>
* </div>
* </div>
*
* #### Popup with checkbox/radio
*
* If you want make popup with list checkbox(or radio) just include checkbox (radio) to popup and add class *ui-popup-checkbox-label* to popup element.
*
* @example
* <div id="listBoxPopup" class="ui-popup">
* <div class="ui-popup-header">When?</div>
* <div class="ui-popup-content" style="height:243px; overflow-y:scroll">
* <ul class="ui-listview">
* <li>
* <label for="check-1" class="ui-popup-checkbox-label">Yesterday</label>
* <input type="checkbox" name="checkset" id="check-1" />
* </li>
* <li>
* <label for="check-2" class="ui-popup-checkbox-label">Today</label>
* <input type="checkbox" name="checkset" id="check-2" />
* </li>
* <li>
* <label for="check-3" class="ui-popup-checkbox-label">Tomorrow</label>
* <input type="checkbox" name="checkset" id="check-3" />
* </li>
* </ul>
* <ul class="ui-listview">
* <li>
* <label for="radio-1" class="ui-popup-radio-label">Mandatory</label>
* <input type="radio" name="radioset" id="radio-1" />
* </li>
* <li>
* <label for="radio-2" class="ui-popup-radio-label">Optional</label>
* <input type="radio" name="radioset" id="radio-2" />
* </li>
* </ul>
* </div>
* <div class="ui-popup-footer">
* <button id="listBoxPopup-close" class="ui-btn">Close</button>
* </div>
* </div>
* </div>
*
* #### Popup with no header and footer
*
* @example
* <div id="listNoTitleNoBtnPopup" class="ui-popup">
* <div class="ui-popup-content" style="height:294px; overflow-y:scroll">
* <ul class="ui-listview">
* <li><a href="">Ringtones 1</a></li>
* <li><a href="">Ringtones 2</a></li>
* <li><a href="">Ringtones 3</a></li>
* </ul>
* </div>
* </div>
*
* #### Toast popup
*
* @example
* <div id="PopupToast" class="ui-popup ui-popup-toast">
* <div class="ui-popup-content">Saving contacts to sim on Samsung</div>
* </div>
*
* ### Create Option popup
*
* Popup inherits value of option positionTo from property data-position-to set in link.
*
* @example
* <!--definition of link, which opens popup and sets its position-->
* <a href="#popupOptionText" data-rel="popup" data-position-to="origin">Text</a>
* <!--definition of popup, which inherites property position from link-->
* <div id="popupOptionText" class="ui-popup">
* <div class="ui-popup-content">
* <ul class="ui-listview">
* <li><a href="#">Option 1</a></li>
* <li><a href="#">Option 2</a></li>
* <li><a href="#">Option 3</a></li>
* <li><a href="#">Option 4</a></li>
* </ul>
* </div>
* </div>
*
* ### Opening and closing popup
*
* To open popup from "a" link using html markup, use the following code:
*
* @example
* <div class="ui-page">
* <header class="ui-header">
* <h2 class="ui-title">Call menu</h2>
* </header>
* <div class="ui-content">
* <a href="#popup" class="ui-btn" data-rel="popup" >Open Popup</a>
* </div>
*
* <div id="popup" class="ui-popup">
* <div class="ui-popup-header">Power saving mode</div>
* <div class="ui-popup-content">
* Turning on Power
* saving mode will
* limit the maximum
* per
* </div>
* <div class="ui-popup-footer">
* <button id="cancel" class="ui-btn">Cancel</button>
* </div>
* </div>
*
* To open the popup widget from JavaScript use method *tau.openPopup(to)*
*
* @example
* tau.openPopup("popup")
*
* To close the popup widget from JavaScript use method *tau.openPopup(to)*
*
* @example
* tau.closePopup("popup")
*
* To find the currently active popup, use the ui-popup-active class.
*
* To bind the popup to a button, use the following code:
*
* @example
* <!--HTML code-->
* <div id="1btnPopup" class="ui-popup">
* <div class="ui-popup-header">Power saving mode</div>
* <div class="ui-popup-content">
* </div>
* <div class="ui-popup-footer">
* <button id="1btnPopup-cancel" class="ui-btn">Cancel</button>
* </div>
* </div>
* <script>
* // Popup opens with button click
* var button = document.getElementById("button");
* button.addEventListener("click", function() {
* tau.openPopup("#1btnPopup");
* });
*
* // Popup closes with Cancel button click
* document.getElementById("1btnPopup-cancel").addEventListener("click", function() {
* tau.closePopup();
* });
* </script>
*
* ## Manual constructor
* For manual creation of popup widget you can use constructor of widget from **tau** namespace:
*
* @example
* var popupElement = document.getElementById("popup"),
* popup = tau.widget.popup(buttonElement);
*
* Constructor has one require parameter **element** which are base **HTMLElement** to create widget. We recommend get this element by method *document.getElementById*.
*
* ## Options for Popup Widget
*
* Options for widget can be defined as _data-..._ attributes or give as parameter in constructor.
*
* You can change option for widget using method **option**.
*
* ## Methods
*
* To call method on widget you can use tau API:
*
* @example
* var popupElement = document.getElementById("popup"),
* popup = tau.widget.popup(buttonElement);
*
* popup.methodName(methodArgument1, methodArgument2, ...);
*
* ## Transitions
*
* By default, the framework doesn't apply transition. To set a custom transition effect, add the data-transition attribute to the link.
*
* @example
* <a href="index.html" data-rel="popup" data-transition="slideup">I'll slide up</a>
*
* Global configuration:
*
* @example
* gear.ui.defaults.popupTransition = "slideup";
*
* ### Transitions list
*
* - **none** Default value, no transition.
* - **slideup** Makes the content of the pop-up slide up.
*
* ## Handling Popup Events
*
* To use popup events, use the following code:
*
* @example
* <!--Popup html code-->
* <div id="popup" class="ui-popup">
* <div class="ui-popup-header"></div>
* <div class="ui-popup-content"></div>
* </div>
* </div>
* <script>
* // Use popup events
* var popup = document.getElementById("popup");
* popup.addEventListener("popupbeforecreate", function() {
* // Implement code for popupbeforecreate event
* });
* </script>
*
* Full list of available events is in [events list section](#events-list).
*
* @author Hyunkook Cho <hk0713.cho@samsung.com>
* @class ns.widget.core.Popup
* @extends ns.widget.core.ContextPopup
*/
(function (window, document, ns) {
"use strict";
var CorePopup = ns.widget.core.ContextPopup,
CorePopupPrototype = CorePopup.prototype,
engine = ns.engine,
objectUtils = ns.util.object,
domUtils = ns.util.DOM,
defaults = {
fullSize: false,
enablePopupScroll: false
},
classes = objectUtils.merge({}, CorePopup.classes, {
popupScroll: "ui-scroll-on",
fixed: "ui-fixed",
sideButton: "ui-side-button",
hasSideButtons: "ui-has-side-buttons",
toast: "ui-popup-toast",
ctx: "ui-ctxpopup"
}),
Popup = function () {
var self = this;
CorePopup.call(self);
self.options = objectUtils.merge(self.options, {
fullSize: ns.getConfig("popupFullSize", defaults.fullSize),
enablePopupScroll: ns.getConfig("enablePopupScroll", defaults.enablePopupScroll)
});
},
prototype = new CorePopup();
/**
* Layouting popup structure
* @method layout
* @member ns.widget.wearable.Popup
*/
prototype._layout = function (element) {
var self = this,
elementClassList = element.classList,
ui = self._ui,
wrapper = ui.wrapper,
header = ui.header,
footer = ui.footer,
content = ui.content,
headerHeight = 0,
footerHeight = 0;
self._blockPageScroll();
CorePopupPrototype._layout.call(self, element);
if (self.options.enablePopupScroll === true) {
element.classList.add(classes.popupScroll);
} else {
element.classList.remove(classes.popupScroll);
}
if (elementClassList.contains(classes.popupScroll)) {
elementClassList.add(classes.build);
if (header) {
headerHeight = header.offsetHeight;
if (header.classList.contains(classes.fixed)) {
content.style.marginTop = headerHeight + "px";
}
}
if (footer) {
footerHeight = footer.offsetHeight;
if (footer.classList.contains(classes.fixed)) {
content.style.marginBottom = footerHeight + "px";
}
if (footer.classList.contains(classes.sideButton)) {
elementClassList.add(classes.hasSideButtons);
}
}
wrapper.style.height = Math.min(content.offsetHeight + headerHeight + footerHeight, element.offsetHeight) + "px";
elementClassList.remove(classes.build);
}
if (self.options.fullSize && !elementClassList.contains(classes.toast) && !elementClassList.contains(classes.ctx)) {
wrapper.style.height = window.innerHeight + "px";
}
};
/**
* Hide popup.
* @method _onHide
* @protected
* @member ns.widget.wearable.Popup
*/
prototype._onHide = function() {
var self = this,
ui = self._ui,
wrapper = ui.wrapper;
wrapper.removeAttribute("style");
self._unblockPageScroll();
CorePopupPrototype._onHide.call(self);
};
prototype._blockPageScroll = function() {
var page = ns.widget.Page(this._ui.page);
if (page.getScroller) {
page.getScroller().style.overflow = "hidden";
}
};
prototype._unblockPageScroll = function() {
var page = ns.widget.Page(this._ui.page);
if (page.getScroller) {
page.getScroller().style.overflow = "";
}
};
Popup.prototype = prototype;
ns.widget.wearable.Popup = Popup;
engine.defineWidget(
"Popup",
"[data-role='popup'], .ui-popup",
[
"open",
"close",
"reposition"
],
Popup,
"wearable",
true
);
}(window, window.document, ns));
/*global define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Router
* Namespace for routers
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Krzysztof Antoszek <k.antoszek@samsung.com>
* @class ns.router
*/
(function (ns) {
"use strict";
ns.router = ns.router || {};
}(ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #History
* Object controls history changes.
*
* @class ns.router.history
* @author Maciej Urbanski <m.urbanski@samsung.com>
*/
(function (window, ns) {
"use strict";
var historyVolatileMode,
object = ns.util.object,
historyUid = 0,
historyActiveIndex = 0,
windowHistory = window.history,
history = {
/**
* Property contains active state in history.
* @property {Object} activeState
* @static
* @member ns.router.history
*/
activeState : null,
/**
* This method replaces or pushes state to history.
* @method replace
* @param {Object} state The state object
* @param {string} stateTitle The title of state
* @param {string} url The new history entry's URL
* @static
* @member ns.router.history
*/
replace: function (state, stateTitle, url) {
var newState = object.merge({}, state, {
uid: historyVolatileMode ? historyActiveIndex : ++historyUid,
stateUrl: url,
stateTitle: stateTitle
});
windowHistory[historyVolatileMode ? "replaceState" : "pushState"](newState, stateTitle, url);
history.setActive(newState);
},
/**
* This method moves backward through history.
* @method back
* @static
* @member ns.router.history
*/
back: function () {
windowHistory.back();
},
/**
* This method sets active state.
* @method setActive
* @param {Object} state Activated state
* @static
* @member ns.router.history
*/
setActive: function (state) {
if (state) {
history.activeState = state;
historyActiveIndex = state.uid;
if (state.volatileRecord) {
history.enableVolatileRecord();
return;
}
}
history.disableVolatileMode();
},
/**
* This method returns "back" if state is in history or "forward" if it is new state.
* @method getDirection
* @param {Object} state Checked state
* @return {"back"|"forward"}
* @static
* @member ns.router.history
*/
getDirection: function (state) {
if (state) {
return state.uid < historyActiveIndex ? "back" : "forward";
}
return "back";
},
/**
* This method sets volatile mode to true.
* @method enableVolatileRecord
* @static
* @member ns.router.history
*/
enableVolatileRecord: function () {
historyVolatileMode = true;
},
/**
* This method sets volatile mode to false.
* @method disableVolatileMode
* @static
* @member ns.router.history
*/
disableVolatileMode: function () {
historyVolatileMode = false;
}
};
ns.router.history = history;
}(window, ns));
/*global window, define */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Drawer Widget
* Core Drawer widget is a base for creating Drawer widgets for profiles. It
* provides drawer functionality - container with ability to open and close with
* an animation.
*
* ##Positioning Drawer left / right
* To change position of a Drawer please set data-position attribute of Drawer
* element to:
* - left (left position, default)
* - right (right position)
*
* ##Opening / Closing Drawer
* To open / close Drawer one can use open() and close() methods.
*
* ##Checking if Drawer is opened.
* To check if Drawer is opened use widget`s isOpen() method.
*
* ##Creating widget
* Core drawer is a base class - examples of creating widgets are described in
* documentation of profiles
*
* @class ns.widget.core.Drawer
* @extends ns.widget.BaseWidget
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
*/
(function (document, ns) {
"use strict";
/**
* @property {Object} Widget Alias for {@link ns.widget.BaseWidget}
* @member ns.widget.core.Drawer
* @private
* @static
*/
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
/**
* @property {Object} selectors Alias for class ns.util.selectors
* @member ns.widget.core.Drawer
* @private
* @static
* @readonly
*/
selectors = ns.util.selectors,
utilDOM = ns.util.DOM,
events = ns.event,
history = ns.router.history,
Gesture = ns.event.gesture,
Page = ns.widget.core.Page,
STATE = {
CLOSED: "closed",
OPENED: "opened",
SLIDING: "sliding",
SETTLING: "settling"
},
CUSTOM_EVENTS = {
OPEN: "draweropen",
CLOSE: "drawerclose"
},
/**
* Default values
*/
DEFAULT = {
WIDTH: 240,
DURATION: 300,
POSITION: "left"
},
/**
* Drawer constructor
* @method Drawer
*/
Drawer = function () {
var self = this;
/**
* Drawer field containing options
* @property options.position {string} Position of Drawer ("left" or "right")
* @property options.width {number} Width of Drawer
* @property options.duration {number} Duration of Drawer entrance animation
* @property options.closeOnClick {boolean} If true Drawer will be closed on overlay
* @property options.overlay {boolean} Sets whether to show an overlay when Drawer is open.
* @property options.drawerTarget {string} Set drawer target element as the css selector
* @property options.enable {boolean} Enable drawer component
* @property options.dragEdge {number} Set the area that can open the drawer as drag gesture in drawer target element
*/
self.options = {
position : DEFAULT.POSITION,
width : DEFAULT.WIDTH,
duration : DEFAULT.DURATION,
closeOnClick: true,
overlay: true,
drawerTarget: "." + Page.classes.uiPage,
enable: true,
dragEdge: 1
};
self._pageSelector = null;
self._isDrag = false;
self._state = STATE.CLOSED;
self._settlingType = STATE.CLOSED;
self._traslatedX = 0;
self._ui = {};
self._eventBoundElement = null;
self._drawerOverlay = null;
},
/**
* Dictionary object containing commonly used widget classes
* @property {Object} classes
* @member ns.widget.core.Drawer
* @private
* @static
* @readonly
*/
classes = {
page : Page.classes.uiPage,
drawer : "ui-drawer",
left : "ui-drawer-left",
right : "ui-drawer-right",
overlay : "ui-drawer-overlay",
open : "ui-drawer-open",
close : "ui-drawer-close"
},
/**
* {Object} Drawer widget prototype
* @member ns.widget.core.Drawer
* @private
* @static
*/
prototype = new BaseWidget();
Drawer.prototype = prototype;
Drawer.classes = classes;
/**
* Unbind drag events
* @method unbindDragEvents
* @param {Object} self
* @param {HTMLElement} element
* @member ns.widget.core.Drawer
* @private
* @static
*/
function unbindDragEvents(self, element) {
var overlayElement = self._ui.drawerOverlay;
events.disableGesture(element);
events.off(element, "drag dragstart dragend dragcancel swipe swipeleft swiperight vmouseup", self, false);
events.prefixedFastOff(self.element, "transitionEnd", self, false);
events.off(window, "resize", self, false);
if (overlayElement) {
events.off(overlayElement, "vclick", self, false);
}
}
/**
* Bind drag events
* @method bindDragEvents
* @param {Object} self
* @param {HTMLElement} element
* @member ns.widget.core.Drawer
* @private
* @static
*/
function bindDragEvents(self, element) {
var overlayElement = self._ui.drawerOverlay;
self._eventBoundElement = element;
events.enableGesture(
element,
new Gesture.Drag(),
new Gesture.Swipe({
orientation: Gesture.Orientation.HORIZONTAL
})
);
events.on(element, "drag dragstart dragend dragcancel swipe swipeleft swiperight vmouseup", self, false);
events.prefixedFastOn(self.element, "transitionEnd", self, false);
events.on(window, "resize", self, false);
if (overlayElement) {
events.on(overlayElement, "vclick", self, false);
}
}
/**
* Handle events
* @method handleEvent
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype.handleEvent = function (event) {
var self = this;
switch (event.type) {
case "drag":
self._onDrag(event);
break;
case "dragstart":
self._onDragStart(event);
break;
case "dragend":
self._onDragEnd(event);
break;
case "dragcancel":
self._onDragCancel(event);
break;
case "vmouseup":
self._onMouseup(event);
break;
case "swipe":
case "swipeleft":
case "swiperight":
self._onSwipe(event);
break;
case "vclick":
self._onClick(event);
break;
case "transitionend":
case "webkitTransitionEnd":
case "mozTransitionEnd":
case "oTransitionEnd":
case "msTransitionEnd":
self._onTransitionEnd(event);
break;
case "resize":
self._onResize(event);
break;
}
};
/**
* MouseUp event handler
* @method _onMouseup
* @param {Event} event
* @member ns.widget.core.Drawer
* @protected
*/
prototype._onMouseup = function (event) {
var self = this;
if (self._state === STATE.SLIDING) {
self.close();
}
};
/**
* Click event handler
* @method _onClick
* @param {Event} event
* @member ns.widget.core.Drawer
* @protected
*/
prototype._onClick = function (event) {
var self = this;
if (self._state === STATE.OPENED) {
self.close();
}
};
/**
* Resize event handler
* @method _onResize
* @param {Event} event
* @member ns.widget.core.Drawer
* @protected
*/
prototype._onResize = function (event) {
var self = this;
// resize event handler
self._refresh();
};
/**
* webkitTransitionEnd event handler
* @method _onTransitionEnd
* @param {Event} event
* @member ns.widget.core.Drawer
* @protected
*/
prototype._onTransitionEnd = function (event) {
var self = this,
position = self.options.position,
drawerOverlay = self._drawerOverlay;
if (self._state === STATE.SETTLING) {
if (self._settlingType === STATE.OPENED) {
self.trigger(CUSTOM_EVENTS.OPEN, {
position: position
});
self._setActive(true);
self._state = STATE.OPENED;
} else {
self.close();
self.trigger(CUSTOM_EVENTS.CLOSE, {
position: position
});
self._setActive(false);
self._state = STATE.CLOSED;
if (drawerOverlay) {
drawerOverlay.style.visibility = "hidden";
}
}
}
};
/**
* Swipe event handler
* @method _onSwipe
* @protected
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype._onSwipe = function (event) {
var self = this,
direction,
options = self.options;
// Now mobile has two swipe event
if (event.detail) {
direction = event.detail.direction === "left" ? "right" : "left";
} else if (event.type === "swiperight") {
direction = "left";
} else if (event.type === "swipeleft") {
direction = "right";
}
if (options.enable && self._isDrag && options.position === direction) {
self.open();
self._isDrag = false;
}
};
/**
* Dragstart event handler
* @method _onDragStart
* @protected
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype._onDragStart = function (event) {
var self = this;
if (self._state === STATE.OPENED) {
return;
}
if (self.options.enable && !self._isDrag && self._state !== STATE.SETTLING && self._checkSideEdge(event)) {
self._isDrag = true;
} else {
self.close();
}
};
/**
* Drag event handler
* @method _onDrag
* @protected
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype._onDrag = function (event) {
var self = this,
deltaX = event.detail.deltaX,
options = self.options,
translatedX = self._traslatedX,
movedX;
if (options.enable && self._isDrag && self._state !== STATE.SETTLING) {
if (options.position === "left") {
movedX = -options.width + deltaX + translatedX;
if (movedX < 0) {
self._translate(movedX, 0);
}
} else {
movedX = window.innerWidth + deltaX - translatedX;
if (movedX > 0 && movedX > window.innerWidth - options.width) {
self._translate(movedX, 0);
}
}
}
};
/**
* DragEnd event handler
* @method _onDragEnd
* @protected
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype._onDragEnd = function (event) {
var self = this,
options = self.options,
detail = event.detail;
if (options.enable && self._isDrag) {
if (Math.abs(detail.deltaX) > options.width / 2) {
self.open();
} else if (self._state !== STATE.SETTLING) {
self.close();
}
}
self._isDrag = false;
};
/**
* DragCancel event handler
* @method _onDragCancel
* @protected
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype._onDragCancel = function (event) {
var self = this;
if (self.options.enable && self._isDrag) {
self.close();
}
self._isDrag = false;
};
/**
* Drawer translate function
* @method _translate
* @param {number} x
* @param {number} duration
* @member ns.widget.core.Drawer
* @protected
*/
prototype._translate = function (x, duration) {
var self = this,
element = self.element;
if (self._state !== STATE.SETTLING) {
self._state = STATE.SLIDING;
}
if (duration) {
utilDOM.setPrefixedStyle(element, "transition", utilDOM.getPrefixedValue("transform " + duration / 1000 + "s ease-out"));
}
// there should be a helper for this :(
utilDOM.setPrefixedStyle(element, "transform", "translate3d(" + x + "px, 0px, 0px)");
if (self.options.overlay) {
self._setOverlay(x);
}
if (!duration) {
self._onTransitionEnd();
}
};
/**
* Set overlay opacity and visibility
* @method _setOverlay
* @param {number} x
* @member ns.widget.core.Drawer
* @protected
*/
prototype._setOverlay = function (x) {
var self = this,
options = self.options,
overlay = self._ui.drawerOverlay,
overlayStyle = overlay.style,
absX = Math.abs(x),
ratio = options.position === "right" ? absX / window.innerWidth : absX / options.width;
if (ratio < 1) {
overlayStyle.visibility = "visible";
} else {
overlayStyle.visibility = "hidden";
}
overlayStyle.opacity = 1 - ratio;
};
/**
* Set active status in drawer router
* @method _setActive
* @param {boolean} active
* @member ns.widget.core.Drawer
* @protected
*/
prototype._setActive = function (active) {
var self = this,
route = engine.getRouter().getRoute("drawer");
if (active) {
route.setActive(self);
} else {
route.setActive(null);
}
};
/**
* Build structure of Drawer widget
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement} Returns built element
* @member ns.widget.core.Drawer
* @protected
*/
prototype._build = function (element) {
var self = this,
ui = self._ui,
options = self.options,
targetElement;
element.classList.add(classes.drawer);
element.style.top = 0;
targetElement = selectors.getClosestBySelector(element, options.drawerTarget);
if (targetElement) {
targetElement.appendChild(element);
targetElement.style.overflowX = "hidden";
}
if (self.options.overlay) {
ui.drawerOverlay = self._createOverlay(element);
ui.drawerOverlay.style.visibility = "hidden";
}
if (!ui.placeholder) {
ui.placeholder = document.createComment(element.id + "-placeholder");
element.parentNode.insertBefore(ui.placeholder, element);
}
ui.targetElement = targetElement;
return element;
};
/**
* Initialization of Drawer widget
* @method _init
* @param {HTMLElement} element
* @member ns.widget.core.Drawer
* @protected
*/
prototype._init = function (element) {
var self = this,
ui = self._ui;
ui.drawerPage = selectors.getClosestByClass(element, classes.page);
ui.drawerPage.style.overflowX = "hidden";
self._initLayout();
return element;
};
/**
* init Drawer widget layout
* @method _initLayout
* @protected
* @member ns.widget.core.Drawer
*/
prototype._initLayout = function () {
var self = this,
options = self.options,
element = self.element,
elementStyle = element.style,
ui = self._ui,
overlayStyle = ui.drawerOverlay ? ui.drawerOverlay.style : false;
options.width = options.width || ui.targetElement.offsetWidth;
elementStyle.width = options.width + "px";
elementStyle.height = ui.targetElement.offsetHeight + "px";
if (overlayStyle) {
overlayStyle.width = window.innerWidth + "px";
overlayStyle.height = window.innerHeight + "px";
overlayStyle.top = 0;
}
if (options.position === "right") {
element.classList.add(classes.right);
self._translate(window.innerWidth, 0);
} else {
// left or default
element.classList.add(classes.left);
self._translate(-options.width, 0);
}
self._state = STATE.CLOSED;
};
/**
* Provides translation if position is set to right
* @method _translateRight
* @member ns.widget.core.Drawer
* @protected
*/
prototype._translateRight = function () {
var self = this,
options = self.options;
if (options.position === "right") {
// If drawer position is right, drawer should be moved right side
if (self._state === STATE.OPENED) {
// drawer opened
self._translate(window.innerWidth - options.width, 0);
} else {
// drawer closed
self._translate(window.innerWidth, 0);
}
}
};
/**
* Check dragstart event whether triggerred on side edge area or not
* @method _checkSideEdge
* @protected
* @param {Event} event
* @member ns.widget.core.Drawer
*/
prototype._checkSideEdge = function (event) {
var self = this,
detail = event.detail,
eventClientX = detail.pointer.clientX - detail.estimatedDeltaX,
options = self.options,
position = options.position,
boundElement = self._eventBoundElement,
boundElementOffsetWidth = boundElement.offsetWidth,
boundElementRightEdge = boundElement.offsetLeft + boundElementOffsetWidth,
dragStartArea = boundElementOffsetWidth * options.dragEdge;
return ((position === "left" && eventClientX > 0 && eventClientX < dragStartArea) ||
(position === "right" && eventClientX > boundElementRightEdge - dragStartArea &&
eventClientX < boundElementRightEdge));
};
/**
* Refreshes Drawer widget
* @method _refresh
* @member ns.widget.core.Drawer
* @protected
*/
prototype._refresh = function () {
// Drawer layout has been set by parent element layout
var self = this;
self._translateRight();
self._initLayout();
};
/**
* Creates Drawer overlay element
* @method _createOverlay
* @param {HTMLElement} element
* @member ns.widget.core.Drawer
* @protected
*/
prototype._createOverlay = function (element) {
var overlayElement = document.createElement("div");
overlayElement.classList.add(classes.overlay);
element.parentNode.insertBefore(overlayElement, element);
return overlayElement;
};
/**
* Binds events to a Drawer widget
* @method _bindEvents
* @member ns.widget.core.Drawer
* @protected
*/
prototype._bindEvents = function () {
var self = this,
targetElement = self._ui.targetElement;
bindDragEvents(self, targetElement);
};
/**
* Enable Drawer widget
* @method _enable
* @protected
* @member ns.widget.core.Drawer
*/
prototype._enable = function () {
this._oneOption("enable", true);
};
/**
* Disable Drawer widget
* @method _disable
* @protected
* @member ns.widget.core.Drawer
*/
prototype._disable = function () {
this._oneOption("enable", false);
};
/**
* Checks Drawer status
* @method isOpen
* @member ns.widget.core.Drawer
* @return {boolean} Returns true if Drawer is open
*/
prototype.isOpen = function () {
return (this._state === STATE.OPENED);
};
/**
* Opens Drawer widget
* @method open
* @param {number} [duration] Duration for opening, if is not set then method take value from options
* @member ns.widget.core.Drawer
*/
prototype.open = function (duration) {
var self = this,
options = self.options,
drawerClassList = self.element.classList,
drawerOverlay = self._ui.drawerOverlay;
if (self._state !== STATE.OPENED) {
self._state = STATE.SETTLING;
self._settlingType = STATE.OPENED;
duration = duration !== undefined ? duration : options.duration;
if (drawerOverlay) {
drawerOverlay.style.visibility = "visible";
}
drawerClassList.remove(classes.close);
drawerClassList.add(classes.open);
if (options.position === "left") {
self._translate(0, duration);
} else {
self._translate(window.innerWidth - options.width, duration);
}
}
};
/**
* Closes Drawer widget
* @method close
* @param {object} options This value is router options whether reverse or not.
* @param {number} [duration] Duration for closing, if is not set then method take value from options
* @member ns.widget.core.Drawer
*/
prototype.close = function (options, duration) {
var self = this,
reverse = options ? options.reverse : false,
selfOptions = self.options,
drawerClassList = self.element.classList;
if (self._state !== STATE.CLOSED) {
if (!reverse && self._state === STATE.OPENED) {
// This method was fired by JS code or this widget.
history.back();
return;
}
self._state = STATE.SETTLING;
self._settlingType = STATE.CLOSED;
duration = duration !== undefined ? duration : selfOptions.duration;
drawerClassList.remove(classes.open);
drawerClassList.add(classes.close);
if (selfOptions.position === "left") {
self._translate(-selfOptions.width, duration);
} else {
self._translate(window.innerWidth, duration);
}
}
};
/**
* Set Drawer drag handler.
* If developer use handler, drag event is bound at handler only.
* @method setDragHandler
* @param {HTMLElement} element
* @member ns.widget.core.Drawer
*/
prototype.setDragHandler = function (element) {
var self = this;
self.options.dragEdge = 1;
unbindDragEvents(self, self._eventBoundElement);
bindDragEvents(self, element);
};
/**
* Transition Drawer widget.
* This method use only positive integer number.
* @method transition
* @param {number} position
* @member ns.widget.core.Drawer
*/
prototype.transition = function (position) {
var self = this,
options = self.options;
if (options.position === "left"){
self._translate(-options.width + position, options.duration);
} else {
self._translate(options.width - position , options.duration);
}
self._traslatedX = position;
};
/**
* Get state of Drawer widget.
*/
prototype.getState = function () {
return this._state;
};
/**
* Destroys Drawer widget
* @method _destroy
* @member ns.widget.core.Drawer
* @protected
*/
prototype._destroy = function () {
var self = this,
ui = self._ui,
drawerOverlay = ui.drawerOverlay,
placeholder = ui.placeholder,
placeholderParent = placeholder.parentNode,
element = self.element;
placeholderParent.insertBefore(element, placeholder);
placeholderParent.removeChild(placeholder);
if (drawerOverlay) {
drawerOverlay.removeEventListener("vclick", self._onClickBound, false);
}
unbindDragEvents(self, self._eventBoundElement);
ui = null;
};
ns.widget.core.Drawer = Drawer;
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Drawer Widget in Wearable
* The drawer component is a panel that the application's sub layout on the left or right edge of the screen.
* This component is hidden most of the time, but user can be opened as swipe gesture from the edge of the screen or click the element that is added event handler,
* handler has drawer.open() method.
*
* Note!
* We recommend to make handler element.
* Because if you didn't set the handler, handler was set page element automatically.
* If you really want to make handler as the page element, you should notice data-drag-edge or dragEdge option value
* because default value, '1', is whole area of handler element.
*
* ## HTML Examples
*
* @example
* <div id="drawerPage" class="ui-page">
* <header id="contentHeader" class="ui-header">
* <h2 class="ui-title">Drawer</h2>
* </header>
* <div id = "content" class="ui-content">
* Drawer
* </div>
*
* <!-- Drawer Handler -->
* <a id="drawerHandler" href="#Drawer" class="drawer-handler">Drawer Button</a>
* <!-- Drawer Widget -->
* <div id="drawer" class="ui-drawer" data-drawer-target="#drawerPage" data-position="left" data-enable="true" data-drag-edge="1">
* <header class="ui-header">
* <h2 class="ui-title">Left Drawer</h2>
* </header>
* <div class="ui-content">
* <p>CONTENT</p>
* </div>
* </div>
* </div>
*
* ## Manual constructor
*
* @example
* (function() {
* var handler = document.getElementById("drawerHandler"),
* page = document.getElementById("drawerPage"),
* drawerElement = document.querySelector(handler.getAttribute("href")),
* drawer = tau.widget.Drawer(drawerElement);
*
* page.addEventListener( "pagebeforeshow", function() {
* drawer.setDragHandler(handler);
* tau.event.on(handler, "mousedown touchstart", function(e) {
* switch (e.type) {
* case "touchstart":
* case "mousedown":
* // open drawer
* drawer.transition(60);
* }
* }, false);
* })();
*
* ##Drawer state
* Drawer has four state type.
* - "closed" - Drawer closed state.
* - "opened" - Drawer opened state.
* - "sliding" - Drawer is sliding state. This state does not mean that will operate open or close.
* - "settling" - drawer is settling state. 'Settle' means open or close status. So, this state means that drawer is animating for opened or closed state.
*
* ##Drawer positioning
* You can declare to drawer position manually. (Default is left)
*
* If you implement data-position attributes value is 'left', drawer appear from left side.
*
* @example
* <div class="ui-drawer" data-position="left" id="leftdrawer">
*
* - "left" - drawer appear from left side
* - "right" - drawer appear from right side
*
* ##Drawer targeting
* You can declare to drawer target manually. (Default is Page)
*
* If you implement data-drawer-target attribute value at CSS selector type, drawer widget will be appended to target.
*
* @example
* <div class="ui-drawer" data-drawer-target="#drawerPage">
*
* ##Drawer enable
* You can declare for whether drawer gesture used or not. (Default is true)
*
* If you implement data-enable attribute value is 'true', you can use the drawer widget.
* This option can be changed by 'enable' or 'disable' method.
*
* @example
* <div class="ui-drawer" data-enable="true">
*
* ##Drawer drag gesture start point
* You can declare to drag gesture start point. (Default is 1)
*
* If you implement data-drag-edge attribute value is '0.5', you can drag gesture start in target width * 0.5 width area.
*
* @example
* <div class="ui-drawer" data-drag-edge="1">
*
* @class ns.widget.wearable.Drawer
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
*/
(function (document, ns) {
"use strict";
var CoreDrawer = ns.widget.core.Drawer,
engine = ns.engine,
object = ns.util.object,
Drawer = function () {
var self = this;
CoreDrawer.call(self);
},
prototype = new CoreDrawer();
Drawer.prototype = prototype;
/**
* Configure Drawer widget
* @method _configure
* @protected
* @param {HTMLElement} element
* @member ns.widget.wearable.Drawer
*/
prototype._configure = function() {
var self = this;
/**
* Widget options
* @property {number} [options.width=0] If you set width is 0, drawer width will set as the css style.
*/
self.options.width = 0;
};
/**
* Set Drawer drag handler.
* If developer use handler, drag event is bound at handler only.
*
* #####Running example in pure JavaScript:
*
* @example
* <!-- Drawer Handlers -->
* <a id="leftDrawerHandler" href="#leftDrawer" class="drawer-handler">Left Handler</a>
*
* <div id="leftDrawer" class="ui-drawer" data-drawer-target="#drawerSinglePage" data-position="left" data-enable="true" data-drag-edge="1">
* <header class="ui-header">
* <h2 class="ui-title">Left Drawer</h2>
* </header>
* <div id="leftClose" class="ui-content">
* <p>Click Close</p>
* </div>
* </div>
*
* <script>
* var handler = document.getElementById("leftDrawerHandler"),
* drawer = tau.widget.Drawer(document.querySelector(handler.getAttribute("href"));
*
* drawer.setDragHandler(handler);
* </script>
*
* @method setDragHandler
* @public
* @param {Element} element
* @member ns.widget.wearable.Drawer
*/
/**
* Transition Drawer widget.
* This method use only positive integer number.
*
* #####Running example in pure JavaScript:
*
* @example
* <!-- Drawer Handlers -->
* <a id="leftDrawerHandler" href="#leftDrawer" class="drawer-handler">Left Handler</a>
*
* <div id="leftDrawer" class="ui-drawer" data-drawer-target="#drawerSinglePage" data-position="left" data-enable="true" data-drag-edge="1">
* <header class="ui-header">
* <h2 class="ui-title">Left Drawer</h2>
* </header>
* <div id="leftClose" class="ui-content">
* <p>Click Close</p>
* </div>
* </div>
*
* <script>
* var handler = document.getElementById("leftDrawerHandler"),
* drawer = tau.widget.Drawer(document.querySelector(handler.getAttribute("href"));
*
* drawer.Transition(60);
* </script>
*
* @method transition
* @public
* @param {Integer} position
* @member ns.widget.wearable.Drawer
*/
/**
* Open Drawer widget.
*
* #####Running example in pure JavaScript:
*
* @example
* <!-- Drawer Handlers -->
* <a id="leftDrawerHandler" href="#leftDrawer" class="drawer-handler">Left Handler</a>
*
* <div id="leftDrawer" class="ui-drawer" data-drawer-target="#drawerSinglePage" data-position="left" data-enable="true" data-drag-edge="1">
* <header class="ui-header">
* <h2 class="ui-title">Left Drawer</h2>
* </header>
* <div id="leftClose" class="ui-content">
* <p>Click Close</p>
* </div>
* </div>
*
* <script>
* var handler = document.getElementById("leftDrawerHandler"),
* drawer = tau.widget.Drawer(document.querySelector(handler.getAttribute("href"));
*
* drawer.open();
* </script>
*
* @method open
* @public
* @member ns.widget.wearable.Drawer
*/
/**
* Close Drawer widget.
*
* @example
* <!-- Drawer Handlers -->
* <a id="leftDrawerHandler" href="#leftDrawer" class="drawer-handler">Left Handler</a>
*
* <div id="leftDrawer" class="ui-drawer" data-drawer-target="#drawerSinglePage" data-position="left" data-enable="true" data-drag-edge="1">
* <header class="ui-header">
* <h2 class="ui-title">Left Drawer</h2>
* </header>
* <div id="leftClose" class="ui-content">
* <p>Click Close</p>
* </div>
* </div>
*
* <script>
* var handler = document.getElementById("leftDrawerHandler"),
* drawer = tau.widget.Drawer(document.querySelector(handler.getAttribute("href"));
*
* drawer.close();
* </script>
*
* @method close
* @public
* @member ns.widget.wearable.Drawer
*/
/**
* Refresh Drawer widget.
* @method refresh
* @protected
* @member ns.widget.wearable.Drawer
*/
/**
* Get state of Drawer widget.
*
* @example
* <!-- Drawer Handlers -->
* <a id="leftDrawerHandler" href="#leftDrawer" class="drawer-handler">Left Handler</a>
*
* <div id="leftDrawer" class="ui-drawer" data-drawer-target="#drawerSinglePage" data-position="left" data-enable="true" data-drag-edge="1">
* <header class="ui-header">
* <h2 class="ui-title">Left Drawer</h2>
* </header>
* <div id="leftClose" class="ui-content">
* <p>Click Close</p>
* </div>
* </div>
*
* <script>
* var handler = document.getElementById("leftDrawerHandler"),
* drawer = tau.widget.Drawer(document.querySelector(handler.getAttribute("href")),
* state;
*
* state = drawer.getState();
* </script>
* @method getState
* @return {String} Drawer state {"closed"|"opened"|"sliding"|"settling"}
* @public
* @member ns.widget.wearable.Drawer
*/
ns.widget.wearable.Drawer = Drawer;
engine.defineWidget(
"Drawer",
".ui-drawer",
[
"transition",
"setDragHandler",
"open",
"close",
"isOpen",
"getState"
],
Drawer,
"wearable"
);
}(window.document, ns));
/*global window, define, ns */
/*jslint nomen: true */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* #Slider
* The slider component changes the range-type browser input to sliders.
*
* ##Default selectors
* In default all **INPUT** tags with type equals _range_ and _data-role=slider_ are changed to TAU sliders.
*
* ###HTML Examples
*
* @example
* <input type="range" name="slider-1" id="slider" value="60" min="0" max="100">
*
* ###Manual constructor
* For manual creation of slider widget you can use constructor of widget
*
* @example
* <input id="slider">
* <script>
* var sliderElement = document.getElementById("slider"),
* slider;
*
* slider = tau.widget.Slider(sliderElement);
*
* // You can make slider component for TizenSlider component name,
* // for example, tau.widget.TizenSlider(sliderElement).
* // But, TizenSlider component name will be deprecated since tizen 2.4
* // because we don't recommend this method.
* </script>
*
* @class ns.widget.core.Slider
* @extends ns.widget.BaseWidget
* @author Hyeoncheol Choi <hc7.choi@samsung.com>
*/
(function (document, ns) {
"use strict";
/**
* @property {Object} Widget Alias for {@link ns.widget.BaseWidget}
* @member ns.widget.core.Drawer
* @private
* @static
*/
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
selectors = ns.util.selectors,
utilDOM = ns.util.DOM,
events = ns.event,
Gesture = ns.event.gesture,
DEFAULT = {
HORIZONTAL: "horizontal"
},
Slider = function () {
var self = this;
/**
* Widget options
* @property {boolean} [options.type="normal"] Slider type. 'normal', 'center' or 'circle'
* @property {string} [options.orientation="horizontal"] Slider orientation. horizontal or vertical
* @property {boolean} [options.expand=false] Slider expand mode. true or false
**/
self.options = {
type: "normal",
orientation: DEFAULT.HORIZONTAL,
expand: false
};
self._ui = {};
},
classes = {
SLIDER: "ui-slider",
SLIDER_HORIZONTAL: "ui-slider-horizontal",
SLIDER_VERTICAL: "ui-slider-vertical",
SLIDER_VALUE: "ui-slider-value",
SLIDER_HANDLER: "ui-slider-handler",
SLIDER_HANDLER_EXPAND: "ui-slider-handler-expand",
SLIDER_CENTER: "ui-slider-center",
SLIDER_HANDLER_ACTIVE: "ui-slider-handler-active"
},
prototype = new BaseWidget();
Slider.prototype = prototype;
Slider.classes = classes;
/**
* Bind events
* @method bindEvents
* @param {Object} self
* @member ns.widget.core.Slider
* @private
* @static
*/
function bindEvents(self) {
var element = self._ui.barElement;
events.enableGesture(
element,
new Gesture.Drag({
orientation: self.options.orientation,
threshold: 0
})
);
events.on(element, "dragstart drag dragend dragcancel", self, false);
}
/**
* unBind events
* @method unbindEvents
* @param {Object} self
* @member ns.widget.core.Slider
* @private
* @static
*/
function unbindEvents(self) {
var element = self._ui.barElement;
events.disableGesture(element);
events.off(element, "dragstart drag dragend dragcancel", self, false);
}
/**
* Build structure of Slider component
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement} Returns built element
* @member ns.widget.core.Slider
* @protected
*/
prototype._build = function(element) {
var self = this,
ui = self._ui,
barElement = document.createElement("div"),
valueElement = document.createElement("div"),
handlerElement = document.createElement("div");
element.style.display = "none";
barElement.classList.add(classes.SLIDER);
valueElement.classList.add(classes.SLIDER_VALUE);
barElement.appendChild(valueElement);
handlerElement.classList.add(classes.SLIDER_HANDLER);
barElement.appendChild(handlerElement);
element.parentNode.appendChild(barElement);
ui.valueElement = valueElement;
ui.handlerElement = handlerElement;
ui.barElement = barElement;
return element;
};
/**
* init Slider component
* @method _init
* @param {HTMLElement} element
* @return {HTMLElement} Returns built element
* @member ns.widget.core.Slider
* @protected
*/
prototype._init = function(element) {
var self = this,
attrMin = parseInt(element.getAttribute("min"), 10),
attrMax = parseInt(element.getAttribute("max"), 10),
attrValue = parseInt(element.getAttribute("value"), 10);
self._min = attrMin ? attrMin : 0;
self._max = attrMax ? attrMax : 100;
self._value = attrValue ? attrValue : self.element.value;
self._interval = self._max - self._min;
self._previousValue = self._value;
self._initLayout();
return element;
};
/**
* init layout of Slider component
* @method _initLayout
* @member ns.widget.core.Slider
* @protected
*/
prototype._initLayout = function() {
var self = this,
options = self.options,
ui = self._ui,
barElement = ui.barElement,
handlerElement = ui.handlerElement;
if (options.orientation === DEFAULT.HORIZONTAL) {
barElement.classList.remove(classes.SLIDER_VERTICAL);
barElement.classList.add(classes.SLIDER_HORIZONTAL);
} else {
barElement.classList.remove(classes.SLIDER_HORIZONTAL);
barElement.classList.add(classes.SLIDER_VERTICAL);
}
options.type === "center" ? barElement.classList.add(classes.SLIDER_CENTER) : barElement.classList.remove(classes.SLIDER_CENTER);
options.expand ? handlerElement.classList.add(classes.SLIDER_HANDLER_EXPAND) : handlerElement.classList.remove(classes.SLIDER_HANDLER_EXPAND);
self._barElementWidth = ui.barElement.offsetWidth;
if (self.options.orientation !== DEFAULT.HORIZONTAL) {
self._barElementHeight = ui.barElement.offsetHeight;
}
self._setValue(self._value);
};
/**
* Set value of Slider center mode
* @method _setCenterValue
* @param {number} value
* @member ns.widget.core.Slider
* @protected
*/
prototype._setCenterValue = function(value) {
var self = this,
ui = self._ui,
validValue,
valueElementValidStyle,
handlerElementValidStyle,
center, validStyle, inValidStyle;
if (self.options.orientation === DEFAULT.HORIZONTAL) {
center = self._barElementWidth / 2;
validValue = self._barElementWidth * (value - self._min) / self._interval;
validStyle = validValue < center ? "right" : "left";
inValidStyle = validValue < center ? "left" : "right";
valueElementValidStyle = "width";
handlerElementValidStyle = "left";
} else {
center = self._barElementHeight / 2;
validValue = self._barElementHeight * (value - self._min) / self._interval;
validStyle = validValue < center ? "bottom" : "top";
inValidStyle = validValue < center ? "top" : "bottom";
valueElementValidStyle = "height";
handlerElementValidStyle = "top";
}
ui.valueElement.style[validStyle] = "50%";
ui.valueElement.style[inValidStyle] = "initial";
ui.valueElement.style[valueElementValidStyle] = Math.abs(center - validValue) + "px";
ui.handlerElement.style[handlerElementValidStyle] = validValue + "px";
};
/**
* Set value of Slider normal mode
* @method _setNormalValue
* @param {number} value
* @member ns.widget.core.Slider
* @protected
*/
prototype._setNormalValue = function(value) {
var self = this,
ui = self._ui,
options = self.options,
barElementLength,
valueElementValidStyle,
handlerElementValidStyle,
validValue;
if (options.orientation === DEFAULT.HORIZONTAL) {
barElementLength = self._barElementWidth;
valueElementValidStyle = "width";
handlerElementValidStyle = "left";
} else {
barElementLength = self._barElementHeight;
valueElementValidStyle = "height";
handlerElementValidStyle = "top";
}
validValue = barElementLength * (value - self._min) / self._interval;
ui.valueElement.style[valueElementValidStyle] = validValue + "px";
ui.handlerElement.style[handlerElementValidStyle] = validValue + "px";
};
/**
* Set value of Slider
* @method _setValue
* @param {number} value
* @member ns.widget.core.Slider
* @protected
*/
prototype._setValue = function(value) {
var self = this,
ui = self._ui,
options = self.options,
element = self.element,
intValue;
if (value < self._min) {
value = self._min;
} else if (value > self._max) {
value = self._max;
}
intValue = parseInt(value, 10);
if (options.type === "center") {
self._setCenterValue(value);
} else if (options.type === "normal") {
self._setNormalValue(value);
}
if (element.value - 0 !== intValue) {
element.setAttribute("value", intValue);
element.value = intValue;
self._value = intValue;
if (self.options.expand) {
ui.handlerElement.innerText = intValue;
}
events.trigger(element, "input");
}
};
/**
* Bind events to Slider
* @method _bindEvents
* @member ns.widget.core.Slider
* @protected
*/
prototype._bindEvents = function() {
bindEvents(this);
};
/**
* Bind event handlers
* @method handleEvent
* @param {Event} event
* @member ns.widget.core.Slider
* @protected
*/
prototype.handleEvent = function(event) {
var self = this;
switch (event.type) {
case "dragstart":
self._onDragstart(event);
break;
case "dragend":
case "dragcancel":
self._onDragend(event);
break;
case "drag":
self._onDrag(event);
break;
}
};
/**
* Drag event handler
* @method _onDrag
* @param {Event} event
* @member ns.widget.core.Slider
* @protected
*/
prototype._onDrag = function(event) {
var self = this,
ui = self._ui,
validPosition,
value;
if (self._active) {
validPosition = self.options.orientation === DEFAULT.HORIZONTAL ?
event.detail.estimatedX - ui.barElement.offsetLeft :
event.detail.estimatedY - utilDOM.getElementOffset(ui.barElement).top + selectors.getScrollableParent(self.element).scrollTop;
value = self.options.orientation === DEFAULT.HORIZONTAL ?
self._interval * validPosition / self._barElementWidth :
self._interval * validPosition / self._barElementHeight;
value += self._min;
self._setValue(value);
}
};
/**
* DragStart event handler
* @method _onDragstart
* @param {Event} event
* @member ns.widget.core.Slider
* @protected
*/
prototype._onDragstart = function(event) {
var self = this,
ui = self._ui,
validPosition = self.options.orientation === DEFAULT.HORIZONTAL ?
event.detail.estimatedX - ui.barElement.offsetLeft :
event.detail.estimatedY - utilDOM.getElementOffset(ui.barElement).top + selectors.getScrollableParent(self.element).scrollTop,
value = self.options.orientation === DEFAULT.HORIZONTAL ?
self._interval * validPosition / self._barElementWidth :
self._interval * validPosition / self._barElementHeight;
ui.handlerElement.classList.add(classes.SLIDER_HANDLER_ACTIVE);
value += self._min;
self._setValue(value);
self._active = true;
};
/**
* DragEnd event handler
* @method _onDragend
* @param {Event} event
* @member ns.widget.core.Slider
* @protected
*/
prototype._onDragend = function() {
var self = this,
ui = self._ui;
ui.handlerElement.classList.remove(classes.SLIDER_HANDLER_ACTIVE);
self._active = false;
if (self._previousValue !== self.element.value) {
events.trigger(self.element, "change");
}
self._previousValue = self.element.value;
};
/**
* Get or Set value of Slider
* @method value
* @param {Number} value
* @return {Number} value
* @member ns.widget.core.Slider
* @protected
*/
prototype.value = function(value) {
var self = this;
if (value !== undefined) {
self._setValue(value);
}
self._previousValue = self.element.value;
return self.element.getAttribute("value");
};
/**
* Refresh to Slider component
* @method refresh
* @member ns.widget.core.Slider
* @protected
*/
prototype.refresh = function() {
this._initLayout();
};
/**
* Destroy Slider component
* @method _destroy
* @member ns.widget.core.Slider
* @protected
*/
prototype._destroy = function() {
var self = this,
barElement = self._ui.barElement;
unbindEvents(self);
barElement.parentNode.removeChild(barElement);
self._ui = null;
self._options = null;
};
ns.widget.core.Slider = Slider;
engine.defineWidget(
"Slider",
"input[data-role='slider'], input[type='range'], input[data-type='range']",
[
"value"
],
Slider,
"core"
);
}(window.document, ns));
/*global window, ns, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Circle Progress Widget
* Shows a control that indicates the progress percentage of an on-going operation by circular shape.
*
* The circle progress widget shows a control that indicates the progress percentage of an on-going operation. This widget can be scaled to be fit inside a parent container.
*
* ### Simple progress bar
* If you don't make any widget "circleprogress" with <progress> element, you can show default progress style.
* To add a circular shape(page size) progressbar in your application, you have to declare <progress> tag in "ui-page" element.
* To add a CircleProgressBar widget to the application, use the following code:
*
* @example
* <div class="ui-page" id="pageCircleProgressBar">
* <header class="ui-header"></header>
* <div class="ui-content"></div>
* <progress class="ui-circle-progress" id="circleprogress" max="20" value="2"></progress>
* </div>
* <script>
* (function(){
*
* var page = document.getElementById( "pageCircleProgressBar" ),
* progressBar = document.getElementById("circleprogress"),
* progressBarWidget;
*
* page.addEventListener( "pageshow", function() {
* var i=0;
* // make Circle Progressbar object
* progressBarWidget = new tau.widget.CircleProgressBar(progressBar);
*
* });
*
* page.addEventListener( "pagehide", function() {
* // release object
* progressBarWidget.destroy();
* });
* }());
* </script>
*
*
* @class ns.widget.wearable.CircleProgressBar
* @since 2.3
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
utilEvent = ns.event,
doms = ns.util.DOM,
eventType = {
/**
* Triggered when the section is changed.
* @event progresschange
* @member ns.widget.wearable.CircleProgressBar
*/
CHANGE: "progresschange"
},
CircleProgressBar = function () {
var self = this,
ui = {};
ui.progressContainer = null;
ui.progressValue = null;
ui.progressValueLeft = null;
ui.progressValueRight = null;
ui.progressValueBg = null;
self.options = {};
self._ui = ui;
self._maxValue = null;
self._value = null;
},
prototype = new BaseWidget(),
CLASSES_PREFIX = "ui-progressbar",
classes = {
uiProgressbar: CLASSES_PREFIX,
uiProgressbarBg: CLASSES_PREFIX + "-bg",
uiProgressbarValue: CLASSES_PREFIX + "-value",
uiProgressbarValueLeft: CLASSES_PREFIX + "-value-left",
uiProgressbarValueRight: CLASSES_PREFIX + "-value-right",
uiProgressbarHalf: CLASSES_PREFIX + "-half"
},
selectors = {
progressContainer: "." + classes.uiProgressbar,
progressBg: "." + classes.uiProgressbarBg,
progressValue: "." + classes.uiProgressbarValue,
progressValueLeft: "." + classes.uiProgressbarValueLeft,
progressValueRight: "." + classes.uiProgressbarValueRight
},
size = {
FULL: "full",
LARGE: "large",
MEDIUM: "medium",
SMALL: "small"
};
CircleProgressBar.classes = classes;
/* make widget refresh with new value */
function refreshProgressBar (self, value) {
var percentValue = value / self._maxValue * 100,
rotateValue,
ui = self._ui;
if (percentValue >= 50) {
ui.progressValue.classList.add(classes.uiProgressbarHalf);
} else {
ui.progressValue.classList.remove(classes.uiProgressbarHalf);
}
rotateValue = 360 * (percentValue/100);
ui.progressValueLeft.style.webkitTransform = "rotate(" + rotateValue + "deg)";
}
function setThicknessStyle (self, value) {
var ui = self._ui;
ui.progressValueLeft.style.borderWidth = value +"px";
ui.progressValueRight.style.borderWidth = value +"px";
ui.progressValueBg.style.borderWidth = value +"px";
}
function setProgressBarSize (self, progressSize) {
var sizeToNumber = parseFloat(progressSize),
ui = self._ui;
if (!isNaN(sizeToNumber)) {
ui.progressContainer.style.fontSize = progressSize + "px";
ui.progressContainer.style.width = progressSize + "px";
ui.progressContainer.style.height = progressSize + "px";
} else {
switch(progressSize) {
case size.FULL:
case size.LARGE:
case size.MEDIUM:
case size.SMALL:
ui.progressContainer.classList.add(CLASSES_PREFIX + "-" + progressSize);
break;
}
ui.progressContainer.style.fontSize = doms.getCSSProperty(ui.progressContainer, "width", 0, "float") + "px";
}
}
function checkOptions (self, option) {
if (option.thickness) {
setThicknessStyle(self, option.thickness);
}
if (option.size) {
setProgressBarSize(self, option.size);
}
if (option.containerClassName) {
self._ui.progressContainer.classList.add(option.containerClassName);
}
}
prototype._configure = function () {
/**
* Options for widget
* @property {Object} options Options for widget
* @property {number} [options.thickness=null] Sets the border width of CircleProgressBar.
* @property {number|"full"|"large"|"medium"|"small"} [options.size="full"] Sets the size of CircleProgressBar.
* @property {string} [options.containerClassName=null] Sets the class name of CircleProgressBar container.
* @member ns.widget.wearable.CircleProgressBar
*/
this.options = {
thickness: null,
size: size.MEDIUM,
containerClassName: null
};
};
/**
* Build CircleProgressBar
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.wearable.CircleProgressBar
*/
prototype._build = function (element) {
var self = this,
ui = self._ui,
progressElement = element,
progressbarContainer, progressbarBg, progressbarValue, progressbarValueLeft, progressbarValueRight;
ui.progressContainer = progressbarContainer = document.createElement("div"),
ui.progressValueBg = progressbarBg = document.createElement("div"),
ui.progressValue = progressbarValue = document.createElement("div"),
ui.progressValueLeft = progressbarValueLeft = document.createElement("div"),
ui.progressValueRight = progressbarValueRight = document.createElement("div");
// set classNames of progressbar DOMs.
progressbarContainer.className = classes.uiProgressbar;
progressbarBg.className = classes.uiProgressbarBg;
progressbarValue.className = classes.uiProgressbarValue;
progressbarValueLeft.className = classes.uiProgressbarValueLeft;
progressbarValueRight.className = classes.uiProgressbarValueRight;
// set id for progress container using "container" prefix
progressbarContainer.id = progressElement.id? progressElement.id + "-container" : "";
progressbarValue.appendChild(progressbarValueLeft);
progressbarValue.appendChild(progressbarValueRight);
progressbarContainer.appendChild(progressbarValue);
progressbarContainer.appendChild(progressbarBg);
progressElement.parentNode.appendChild(progressbarContainer);
progressElement.parentNode.insertBefore(progressElement, progressbarContainer);
return element;
};
/**
* Init CircleProgressBar
* @method _init
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.wearable.CircleProgressBar
*/
prototype._init = function (element) {
var self = this,
ui = self._ui,
progressElement = element,
elementParent = element.parentNode,
options = self.options;
ui.progressContainer = ui.progressContainer || elementParent.querySelector(selectors.progressContainer);
ui.progressValueBg = ui.progressValueBg || elementParent.querySelector(selectors.progressValueBg);
ui.progressValue = ui.progressValue || elementParent.querySelector(selectors.progressValue);
ui.progressValueLeft = ui.progressValueLeft || elementParent.querySelector(selectors.progressValueLeft);
ui.progressValueRight = ui.progressValueRight || elementParent.querySelector(selectors.progressValueRight);
self._maxValue = doms.getNumberFromAttribute(progressElement, "max", null, 100);
// max value must be positive number bigger than 0
if (self._maxValue <= 0) {
ns.error("max value of progress must be positive number that bigger than zero!");
self._maxValue = 100;
}
self._value = doms.getNumberFromAttribute(progressElement, "value", null, 50);
checkOptions(self, options);
refreshProgressBar(self, self._value);
return element;
};
/**
* Get or Set value of the widget
*
* Return element value or set the value
*
* @example
* <progress class="ui-circle-progress" id="circleprogress" max="20" value="2"></progress>
* <script>
* var progressbar = document.getElementById("circleprogress"),
progressbarWidget = tau.widget.CircleProgressBar(progressbar),
* // return value in progress tag
* value = progressbarWidget.value();
* // sets the value for the progress
* progressbarWidget.value("15");
* </script>
* @method value
* return {string} In get mode return element value
* @since 2.3
* @member ns.widget.wearable.CircleProgressBar
*/
/**
* Get value of Circle Progressbar
* @method _getValue
* @protected
* @memeber ns.widget.wearable.CircleProgressBar
*/
prototype._getValue = function () {
return this.element.getAttribute("value");
};
/**
* Set value of Circle Progressbar
* @method _setValue
* @param {string} value
* @protected
* @member ns.widget.wearable.CircleProgressBar
*/
prototype._setValue = function (inputValue) {
var self = this,
value,
selfElementValue;
if (inputValue > self._maxValue) {
value = self._maxValue;
} else if (inputValue < 0) {
value = 0;
} else if (isNaN(inputValue)) {
value = 0;
} else {
value = inputValue;
}
doms.setAttribute(self.element, "value", value);
if (self._value !== value) {
self._value = value;
utilEvent.trigger(self.element, eventType.CHANGE);
refreshProgressBar(self, value);
}
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.wearable.CircleProgressBar
*/
prototype._refresh = function () {
var self = this;
self._reset();
checkOptions(self, self.options);
refreshProgressBar(self, self._getValue());
return null;
};
/**
* Reset style of Value elements
* @method _reset
* @protected
* @member ns.widget.wearable.CircleProgressBar
*/
prototype._reset = function () {
var self = this,
ui = self._ui;
ui.progressValue.classList.remove(classes.uiProgressbarHalf);
ui.progressValueLeft.style.webkitTransform = "";
if (self.options.thickness) {
ui.progressValueLeft.style.borderWidth = "";
ui.progressValueRight.style.borderWidth = "";
ui.progressValueBg.style.borderWidth = "";
}
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.wearable.CircleProgressBar
*/
prototype._destroy = function () {
var self = this;
self._reset();
// remove doms
self.element.parentNode.removeChild(self._ui.progressContainer);
// clear variables
self.element = null;
self._ui = null;
self._maxValue = null;
self._value = null;
return null;
};
CircleProgressBar.prototype = prototype;
ns.widget.wearable.CircleProgressBar = CircleProgressBar;
engine.defineWidget(
"CircleProgressBar",
".ui-circle-progress",
[],
CircleProgressBar,
"wearable"
);
}(window.document, ns));
/*global window, define */
/*jslint nomen: true */
/**
* # Slider Widget
* Wearable Slider component has two types, first is normal slider type another is circle slider type.
* Circle slider type has provided to rotary event handling in component side.
* Normal slider type is default type.
*
* ## Default selectors
*
* To add a slider component to the application, use the following code:
*
* @example
* // Normal type
* <input id="circle" data-type="normal" name="circleSlider" type="range" value="20" min="0" max="100" />
*
* // OR Circle type
* <input id="circle" data-type="circle" name="circleSlider" type="range" value="20" min="0" max="100" />
*
* ## JavaScript API
*
* Slider widget hasn't JavaScript API.
* @class ns.widget.wearable.Slider
*/
(function (document, ns) {
"use strict";
var CoreSlider = ns.widget.core.Slider,
CoreSliderPrototype = CoreSlider.prototype,
engine = ns.engine,
events = ns.event,
CirclePB = ns.widget.wearable.CircleProgressBar,
CirclePBPrototype = new CirclePB(),
Slider = function () {
var self = this;
CoreSlider.call(self);
},
prototype = new CoreSlider();
Slider.prototype = prototype;
function bindCircleEvents(self) {
events.on(document, "rotarydetent", self, false);
}
function unbindCircleEvents(self) {
events.off(document, "rotarydetent", self, false);
}
/**
* Configure Slider widget
* @method _configure
* @protected
* @member ns.widget.wearable.Slider
*/
prototype._configure = function() {
var self = this,
options = self.options;
options.size = "full";
};
/**
* Build Slider widget
* @method _build
* @protected
* @param {HTMLElement} element
* @member ns.widget.wearable.Slider
*/
prototype._build = function(element) {
var self = this,
options = self.options;
if (options.type === "circle") {
element.style.display = "none";
CirclePBPrototype._build.call(self, element);
} else {
CoreSliderPrototype._build.call(self, element);
}
return element;
};
/**
* Init Slider widget
* @method _init
* @protected
* @param {HTMLElement} element
* @member ns.widget.wearable.Slider
*/
prototype._init = function(element) {
var self = this,
options = self.options;
if (options.type === "circle") {
CirclePBPrototype._init.call(self, element);
} else {
CoreSliderPrototype._init.call(self, element);
}
return element;
};
/**
* Bind events Slider widget
* @method _bindEvents
* @protected
* @member ns.widget.wearable.Slider
*/
prototype._bindEvents = function() {
var self = this,
options = self.options;
if (options.type === "circle") {
bindCircleEvents(self);
} else {
CoreSliderPrototype._bindEvents.call(self);
}
};
/**
* Bind event handlers
* @method handleEvent
* @param {Event} event
* @member ns.widget.wearable.Slider
* @protected
*/
prototype.handleEvent = function(event) {
var self = this,
options = self.options;
if (options.type === "circle") {
switch (event.type) {
case "rotarydetent":
self._onRotary(event);
break;
}
} else {
CoreSliderPrototype.handleEvent.call(self, event);
}
};
/**
* Rotarydetent event handler
* @method _onRotary
* @param {Event} event
* @member ns.widget.wearable.Slider
* @protected
*/
prototype._onRotary = function(event) {
var self = this,
direction = event.detail.direction,
value = CirclePBPrototype._getValue.call(self);
if (direction === "CW") {
if (value < self._maxValue) {
value++;
} else {
value = self._maxValue;
}
} else if (direction === "CCW") {
if (value > 0) {
value--;
} else {
value = 0;
}
}
CirclePBPrototype._setValue.call(self, value);
};
/**
* Get/set slider value
* @method value
* @param {Number} value
* @member ns.widget.wearable.Slider
* @public
*/
prototype.value = function(value) {
var self = this,
options = self.options,
result;
if (options.type === "circle") {
if (value) {
CirclePBPrototype._setValue.call(self, value);
} else {
result = CirclePBPrototype._getValue.call(self);
}
} else {
result = CoreSliderPrototype.value.call(self, value);
}
if (result) {
return result;
}
};
/**
* Destroy Slider component
* @method _destroy
* @member ns.widget.wearable.Slider
* @protected
*/
prototype._destroy = function () {
var self = this,
options = self.options;
if (options.type === "circle") {
unbindCircleEvents(self);
self._ui = null;
self._options = null;
} else {
CoreSliderPrototype._destroy.call(self);
}
};
ns.widget.wearable.Slider = Slider;
engine.defineWidget(
"Slider",
"input[data-role='slider'], input[type='range'], input[data-type='range']",
[
"value"
],
Slider,
"wearable",
true
);
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Listview Widget
* Shows a list view.
*
* The list widget is used to display, for example, navigation data, results, and data entries. The following table describes the supported list classes.
*
* ## Default selectors
*
* Default selector for listview widget is class *ui-listview*.
*
* To add a list widget to the application, use the following code:
*
* ### List with basic items
*
* You can add a basic list widget as follows:
*
* @example
* <ul class="ui-listview">
* <li>1line</li>
* <li>2line</li>
* <li>3line</li>
* <li>4line</li>
* <li>5line</li>
* </ul>
*
* ### List with link items
*
* You can add a list widget with a link and press effect that allows the user to click each list item as follows:
*
* @example
* <ul class="ui-listview">
* <li>
* <a href="#">1line</a>
* </li>
* <li>
* <a href="#">2line</a>
* </li>
* <li>
* <a href="#">3line</a>
* </li>
* <li>
* <a href="#">4line</a>
* </li>
* <li>
* <a href="#">5line</a>
* </li>
* </ul>
*
* ## JavaScript API
*
* Listview widget hasn't JavaScript API.
*
* @class ns.widget.wearable.Listview
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
Listview = function () {
},
prototype = new BaseWidget();
/**
* Dictionary for listview related events.
* For listview, it is an empty object.
* @property {Object} events
* @member ns.widget.wearable.Listview
* @static
*/
Listview.events = {};
/**
* Build Listview
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.wearable.Listview
*/
prototype._build = function (element) {
return element;
};
prototype._init = function (element) {
return element;
};
prototype._bindEvents = function (element) {
return element;
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.wearable.Listview
*/
prototype._refresh = function () {
return null;
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.wearable.Listview
*/
prototype._destroy = function () {
return null;
};
Listview.prototype = prototype;
ns.widget.wearable.Listview = Listview;
engine.defineWidget(
"Listview",
".ui-listview",
[],
Listview,
"wearable"
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* @author Maciej Urbanski <m.urbanski@samsung.com>
*/
(function (ns) {
"use strict";
/** @namespace ns.widget.wearable */
ns.widget.core.indexscrollbar = ns.widget.core.indexscrollbar || {};
}(ns));
/*global define, ns, document, window */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #IndexBar widget
* Widget creates bar with index.
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Jadwiga Sosnowska <j.sosnowska@samsung.com>
* @class ns.widget.wearable.indexscrollbar.IndexBar
*/
(function (document, ns) {
"use strict";
var utilsObject = ns.util.object,
utilsDOM = ns.util.DOM;
function IndexBar(element, options) {
this.element = element;
this.options = utilsObject.merge(options, this._options, false);
this.container = this.options.container;
this.indices = {
original: this.options.index,
merged: []
};
this._init();
return this;
}
IndexBar.prototype = {
_options: {
container: null,
offsetLeft: 0,
index: [],
verticalCenter: false,
moreChar: "*",
moreCharLineHeight: 9,
indexHeight: 41,
selectedClass: "ui-state-selected",
ulClass: null,
maxIndexLen : 0
},
_init: function() {
this.indices.original = this.options.index;
this.indexLookupTable = [];
this.indexElements = null;
this.selectedIndex = -1;
this.visiblity = "hidden";
this._setMaxIndexLen();
this._makeMergedIndices();
this._drawDOM();
this._appendToContainer();
if(this.options.verticalCenter) {
this._adjustVerticalCenter();
}
this._setIndexCellInfo();
},
_clear: function() {
while(this.element.firstChild) {
this.element.removeChild(this.element.firstChild);
}
this.indices.merged.length = 0;
this.indexLookupTable.length = 0;
this.indexElements = null;
this.selectedIndex = -1;
this.visiblity = null;
},
/**
* Refreshes widget.
* @method refresh
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
refresh: function() {
this._clear();
this._init();
},
/**
* Destroys widget.
* @method destroy
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
destroy: function() {
this._clear();
},
/**
* Shows widget.
* @method show
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
show: function() {
this.visibility = "visible";
this.element.style.visibility = this.visibility;
},
/**
* Hides widget.
* @method hide
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
hide: function() {
this.visibility = "hidden";
this.element.style.visibility = this.visibility;
},
/**
* Get if the visibility status is shown or not
* @method isShown
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
isShown: function() {
return "visible" === this.visibility;
},
_setMaxIndexLen: function() {
var maxIndexLen,
containerHeight = this.container.offsetHeight;
maxIndexLen = Math.floor( containerHeight / this.options.indexHeight );
if(maxIndexLen > 0 && maxIndexLen%2 === 0) {
maxIndexLen -= 1; // Ensure odd number
}
this.options.maxIndexLen = this.options.maxIndexLen > 0 ? Math.min(maxIndexLen, this.options.maxIndexLen) : maxIndexLen;
},
_makeMergedIndices: function() {
var origIndices = this.indices.original,
origIndexLen = origIndices.length,
visibleIndexLen = Math.min(this.options.maxIndexLen, origIndexLen),
totalLeft = origIndexLen - visibleIndexLen,
nIndexPerItem = parseInt(totalLeft / parseInt(visibleIndexLen/2, 10), 10),
leftItems = totalLeft % parseInt(visibleIndexLen/2, 10),
indexItemSize = [],
mergedIndices = [],
i, len, position=0;
for(i = 0, len = visibleIndexLen; i < len; i++) {
indexItemSize[i] = 1;
if (i % 2) { // even number: omitter
indexItemSize[i] += nIndexPerItem + (leftItems-- > 0 ? 1 : 0);
}
position += indexItemSize[i];
mergedIndices.push( {
start: position-1,
length: indexItemSize[i]
});
}
this.indices.merged = mergedIndices;
},
_drawDOM: function() {
var origIndices = this.indices.original,
indices = this.indices.merged,
indexLen = indices.length,
indexHeight = this.options.indexHeight,
moreChar = this.options.moreChar,
addMoreCharLineHeight = this.options.moreCharLineHeight,
text,
frag,
li,
i,
m;
frag = document.createDocumentFragment();
for(i=0; i < indexLen; i++) {
m = indices[i];
text = m.length === 1 ? origIndices[m.start] : moreChar;
li = document.createElement("li");
li.innerText = text.toUpperCase();
li.style.height = indexHeight + "px";
li.style.lineHeight = text === moreChar ? indexHeight + addMoreCharLineHeight + "px" : indexHeight + "px";
frag.appendChild(li);
}
this.element.appendChild(frag);
if(this.options.ulClass) {
this.element.classList.add( this.options.ulClass );
}
},
_adjustVerticalCenter: function() {
var nItem = this.indices.merged.length,
totalIndexLen = nItem * this.options.indexHeight,
vPadding = parseInt((this.container.offsetHeight - totalIndexLen) / 2, 10);
this.element.style.paddingTop = vPadding + "px";
},
_appendToContainer: function() {
this.container.appendChild(this.element);
this.element.style.left = this.options.offsetLeft + "px";
},
/**
* Sets padding top for element.
* @method setPaddingTop
* @param {number} paddingTop
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
setPaddingTop: function(paddingTop) {
var height = this.element.clientHeight,
oldPaddingTop = this.element.style.paddingTop,
containerHeight = this.container.clientHeight;
if(oldPaddingTop === "") {
oldPaddingTop = 0;
} else {
oldPaddingTop = parseInt(oldPaddingTop, 10);
}
height = height - oldPaddingTop;
if(height > containerHeight) {
paddingTop -= (paddingTop + height - containerHeight);
}
this.element.style.paddingTop = paddingTop + "px";
this._setIndexCellInfo(); // update index cell info
},
/**
* Returns element's offsetTop of given index.
* @method getOffsetTopByIndex
* @param {number} index
* @return {number}
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
getOffsetTopByIndex: function(index) {
var cellIndex = this.indexLookupTable[index].cellIndex,
el = this.indexElements[cellIndex],
offsetTop = el.offsetTop;
return offsetTop;
},
_setIndexCellInfo: function() {
var element = this.element,
mergedIndices = this.indices.merged,
containerOffsetTop = utilsDOM.getElementOffset(this.container).top,
listitems = this.element.querySelectorAll("LI"),
lookupTable = [];
[].forEach.call(listitems, function(node, idx) {
var m = mergedIndices[idx],
i = m.start,
len = i + m.length,
top = containerOffsetTop + node.offsetTop,
height = node.offsetHeight / m.length;
for ( ; i < len; i++ ) {
lookupTable.push({
cellIndex: idx,
top: top,
range: height
});
top += height;
}
});
this.indexLookupTable = lookupTable;
this.indexElements = element.children;
},
/**
* Returns index for given position.
* @method getIndexByPosition
* @param {number} posY
* @return {number}
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
getIndexByPosition: function(posY) {
var table = this.indexLookupTable,
info,
i, len, range;
// boundary check
if( table[0] ) {
info = table[0];
if(posY < info.top) {
return 0;
}
}
if( table[table.length -1] ) {
info = table[table.length -1];
if(posY >= info.top + info.range) {
return table.length - 1;
}
}
for ( i=0, len=table.length; i < len; i++) {
info = table[i];
range = posY - info.top;
if ( range >= 0 && range < info.range ) {
return i;
}
}
return 0;
},
/**
* Returns value for given index.
* @method getValueByIndex
* @param {number} idx
* @return {number}
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
getValueByIndex: function(idx) {
if(idx < 0) { idx = 0; }
return this.indices.original[idx];
},
/**
* Select given index
* @method select
* @param {number} idx
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
select: function(idx) {
var cellIndex,
eCell;
this.clearSelected();
if(this.selectedIndex === idx) {
return;
}
this.selectedIndex = idx;
cellIndex = this.indexLookupTable[idx].cellIndex;
eCell = this.indexElements[cellIndex];
eCell.classList.add(this.options.selectedClass);
},
/**
* Clears selected class.
* @method clearSelected
* @member ns.widget.wearable.indexscrollbar.IndexBar
*/
clearSelected: function() {
var el = this.element,
selectedClass = this.options.selectedClass,
selectedElement = el.querySelectorAll("."+selectedClass);
[].forEach.call(selectedElement, function(node) {
node.classList.remove(selectedClass);
});
this.selectedIndex = -1;
}
};
ns.widget.core.indexscrollbar.IndexBar = IndexBar;
}(window.document, ns));
/*global define, ns, document, window */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #IndexIndicator widget
* Class creates index indicator.
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Jadwiga Sosnowska <j.sosnowska@samsung.com>
* @class ns.widget.wearable.indexscrollbar.IndexIndicator
*/
(function (document, ns) {
"use strict";
var utilsObject = ns.util.object,
events = ns.event;
/**
* @brief block 'unexpected bouncing effect' on indexscroller indicator.
*/
function blockEvent (event) {
event.preventDefault();
event.stopPropagation();
}
function IndexIndicator(element, options) {
this.element = element;
this.options = utilsObject.merge(options, this._options, false);
this.value = null;
this._init();
return this;
}
IndexIndicator.prototype = {
_options: {
className: "ui-indexscrollbar-indicator",
selectedClass: "ui-selected",
container: null
},
_init: function() {
var self = this,
options = self.options,
element = self.element;
element.className = options.className;
element.innerHTML = "<span></span>";
events.on(element, ["touchstart", "touchmove"], blockEvent, false);
// Add to DOM tree
options.referenceElement.parentNode.insertBefore(element, options.referenceElement);
self.fitToContainer();
},
/**
* Fits size to container.
* @method fitToContainer
* @member ns.widget.wearable.indexscrollbar.IndexIndicator
*/
fitToContainer: function() {
var element = this.element,
container = this.options.container,
containerPosition = window.getComputedStyle(container).position;
element.style.width = container.offsetWidth + "px";
element.style.height = container.offsetHeight + "px";
element.style.top = container.offsetTop + "px";
element.style.left = container.offsetLeft + "px";
},
/**
* Sets value of widget.
* @method setValue
* @param {string} value
* @member ns.widget.wearable.indexscrollbar.IndexIndicator
*/
setValue: function( value ) {
this.value = value; // remember value
value = value.toUpperCase();
var selected = value.substr(value.length - 1),
remained = value.substr(0, value.length - 1),
inner = "<span>" + remained + "</span><span class=\"ui-selected\">" + selected + "</span>";
this.element.firstChild.innerHTML = inner; // Set indicator text
},
/**
* Shows widget.
* @method show
* @member ns.widget.wearable.indexscrollbar.IndexIndicator
*/
show: function() {
//this.element.style.visibility="visible";
this.element.style.display="block";
},
/**
* Hides widget.
* @method hide
* @member ns.widget.wearable.indexscrollbar.IndexIndicator
*/
hide: function() {
this.element.style.display="none";
},
/**
* Destroys widget.
* @method destroy
* @member ns.widget.wearable.indexscrollbar.IndexIndicator
*/
destroy: function() {
var element = this.element;
while(element.firstChild) {
element.removeChild(element.firstChild);
}
events.off(element, ["touchstart", "touchmove"], blockEvent, false);
this.element = null; // unreference element
}
};
ns.widget.core.indexscrollbar.IndexIndicator = IndexIndicator;
}(window.document, ns));
/*global define, ns, document, window */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/**
* #IndexScrollbar Widget
* Shows an index scroll bar with indices, usually for the list.
*
* The index scroll bar widget shows on the screen a scrollbar with indices,
* and fires a select event when the index characters are clicked.
* The following table describes the supported index scroll bar APIs.
*
* ## Manual constructor
* For manual creation of widget you can use constructor of widget from **tau** namespace:
*
* @example
* var indexscrollbarElement = document.getElementById('indexscrollbar'),
* indexscrollbar = tau.widget.IndexScrollbar(IndexScrollbar, {index: "A,B,C"});
*
* Constructor has one require parameter **element** which are base **HTMLElement** to create widget.
* We recommend get this element by method *document.getElementById*. Second parameter is **options**
* and it is a object with options for widget.
*
* To add an IndexScrollbar widget to the application, use the following code:
*
* @example
* <div id="foo" class="ui-indexscrollbar" data-index="A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,Z"></div>
* <script>
* (function() {
* var elem = document.getElementById("foo");
* tau.widget.IndexScrollbar(elem);
* elem.addEventListener("select", function( event ) {
* var index = event.detail.index;
* console.log(index);
* });
* }());
* </script>
*
* The index value can be retrieved by accessing event.detail.index property.
*
* In the following example, the list scrolls to the position of the list item defined using
* the li-divider class, selected by the index scroll bar:
*
* @example
* <div id="pageIndexScrollbar" class="ui-page">
* <header class="ui-header">
* <h2 class="ui-title">IndexScrollbar</h2>
* </header>
* <section class="ui-content">
* <div style="overflow-y:scroll;">
* <div id="indexscrollbar1"
* class="ui-indexscrollbar"
* data-index="A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,Z">
* </div>
* <ul class="ui-listview" id="list1">
* <li class="li-divider">A</li>
* <li>Anton</li>
* <li>Arabella</li>
* <li>Art</li>
* <li class="li-divider">B</li>
* <li>Barry</li>
* <li>Bibi</li>
* <li>Billy</li>
* <li>Bob</li>
* <li class="li-divider">D</li>
* <li>Daisy</li>
* <li>Derek</li>
* <li>Desmond</li>
* </ul>
* </div>
* </section>
* <script>
* (function () {
* var page = document.getElementById("pageIndexScrollbar");
* page.addEventListener("pagecreate", function () {
* var elem = document.getElementById("indexscrollbar1"), // Index scroll bar element
* elList = document.getElementById("list1"), // List element
* elDividers = elList.getElementsByClassName("li-divider"), // List items (dividers)
* elScroller = elList.parentElement, // List's parent item (overflow-y:scroll)
* dividers = {}, // Collection of list dividers
* indices = [], // List of index
* elDivider,
* i, idx;
*
* // For all list dividers
* for (i = 0; i < elDividers.length; i++) {
* // Add the list divider elements to the collection
* elDivider = elDividers[i];
* // li element having the li-divider class
* idx = elDivider.innerText;
* // Get a text (index value)
* dividers[idx] = elDivider;
* // Remember the element
*
* // Add the index to the index list
* indices.push(idx);
* }
*
* // Change the data-index attribute to the indexscrollbar element
* // before initializing IndexScrollbar widget
* elem.setAttribute("data-index", indices.join(","));
*
* // Create index scroll bar
* tau.IndexScrollbar(elem);
*
* // Bind the select callback
* elem.addEventListener("select", function (ev) {
* var elDivider,
* idx = ev.detail.index;
* elDivider = dividers[idx];
* if (elDivider) {
* // Scroll to the li-divider element
* elScroller.scrollTop = elDivider.offsetTop - elScroller.offsetTop;
* }
* });
* });
* }());
* </script>
* </div>
*
* The following example uses the supplementScroll argument, which shows a level 2 index scroll bar.
* The application code must contain a level 2 index array for each level 1 index character.
* The example shows a way to analyze list items and create a dictionary (secondIndex) for level 1
* indices for the index scroll bar, and a dictionary (keyItem) for moving list items at runtime:
*
* @example
* <div id="indexscrollbar2" class="ui-indexscrollbar"
* data-index="A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,Z">
* </div>
* <ul class="ui-listview" id="ibar2_list2">
* <li>Anton</li>
* <li>Arabella</li>
* <li>Art</li>
* <li>Barry</li>
* <li>Bibi</li>
* <li>Billy</li>
* <li>Bob</li>
* <li>Carry</li>
* <li>Cibi</li>
* <li>Daisy</li>
* <li>Derek</li>
* <li>Desmond</li>
* </ul>
*
* <script>
* (function () {
* var page = document.getElementById("pageIndexScrollbar2"),
* isb,
* index = [],
* supIndex = {},
* elIndex = {};
* page.addEventListener("pageshow", function () {
* var elisb = document.getElementById("indexscrollbar2"),
* elList = document.getElementById("ibar2_list2"), // List element
* elItems = elList.children,
* elScroller = elList.parentElement, // Scroller (overflow-y:hidden)
* indexData = getIndexData(
* {
* array: elItems,
* getTextValue: function (array, i) {
* return array[i].innerText;
* }
* });
*
* function getIndexData(options) {
* var array = options.array,
* getTextValue = options.getTextValue,
* item,
* text,
* firstIndex = [],
* secondIndex = {},
* keyItem = {},
* c1 = null,
* c2 = null,
* i;
*
* for (i = 0; i < array.length; i++) {
* item = array[i];
* text = getTextValue(array, i);
* if (text.length > 0) {
* if (!c1 || c1 !== text[0]) {
* // New c1
* c1 = text[0];
* firstIndex.push(c1);
* keyItem[c1] = item;
* secondIndex[c1] = [];
* c2 = text[1];
* if (c2) {
* secondIndex[c1].push(c2);
* }
* else {
* c2 = '';
* }
* keyItem[c1 + c2] = item;
* }
* else {
* // Existing c1
* if (c2 !== text[1]) {
* c2 = text[1];
* secondIndex[c1].push(c2);
* keyItem[c1 + c2] = item;
* }
* }
* }
* }
* return {
* firstIndex: firstIndex,
* secondIndex: secondIndex,
* keyItem: keyItem
* };
* }
*
* // Update the data-index attribute to the indexscrollbar element, with the index list above
* elisb.setAttribute("data-index", indexData.firstIndex);
* // Create IndexScrollbar
* isb = new tau.IndexScrollbar(elisb, {
* index: indexData.firstIndex,
* supplementaryIndex: function (firstIndex) {
* return indexData.secondIndex[firstIndex];
* }
* });
* // Bind the select callback
* elisb.addEventListener("select", function (ev) {
* var el,
* index = ev.detail.index;
* el = indexData.keyItem[index];
* if (el) {
* // Scroll to the li-divider element
* elScroller.scrollTop = el.offsetTop - elScroller.offsetTop;
* }
* });
* });
* page.addEventListener("pagehide", function () {
* console.log('isb2:destroy');
* isb.destroy();
* index.length = 0;
* supIndex = {};
* elIndex = {};
* });
* }());
* </script>
*
* ##Options for widget
*
* Options for widget can be defined as _data-..._ attributes or give as parameter in constructor.
*
* You can change option for widget using method **option**.
*
* ##Methods
*
* To call method on widget you can use tau API:
*
* First API is from tau namespace:
*
* @example
* var indexscrollbarElement = document.getElementById('indexscrollbar'),
* indexscrollbar = tau.widget.IndexScrollbar(indexscrollbarElement);
*
* indexscrollbar.methodName(methodArgument1, methodArgument2, ...);
*
* @author Maciej Urbanski <m.urbanski@samsung.com>
* @author Jadwiga Sosnowska <j.sosnowska@samsung.com>
* @author Tomasz Lukawski <t.lukawski@samsung.com>
* @class ns.widget.core.IndexScrollbar
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var IndexScrollbar = function() {
// Support calling without 'new' keyword
this.element = null;
this.indicator = null;
this.indexBar1 = null; // First IndexBar. Always shown.
this.indexBar2 = null; // 2-depth IndexBar. shown if needed.
this.index = null;
this.touchAreaOffsetLeft = 0;
this.indexElements = null;
this.selectEventTriggerTimeoutId = null;
this.ulMarginTop = 0;
this.eventHandlers = {};
},
BaseWidget = ns.widget.BaseWidget,
/**
* Alias for class {@link ns.engine}
* @property {Object} engine
* @member ns.widget.core.IndexScrollbar
* @private
* @static
*/
engine = ns.engine,
/**
* Alias for class {@link ns.event}
* @property {Object} events
* @member ns.widget.core.IndexScrollbar
* @private
* @static
*/
events = ns.event,
/**
* Alias for class {@link ns.util.object}
* @property {Object} utilsObject
* @member ns.widget.core.IndexScrollbar
* @private
* @static
*/
utilsObject = ns.util.object,
/**
* Alias for class ns.util.DOM
* @property {ns.util.DOM} doms
* @member ns.widget.wearable.IndexScrollbar
* @private
* @static
*/
doms = ns.util.DOM,
IndexBar = ns.widget.core.indexscrollbar.IndexBar,
IndexIndicator = ns.widget.core.indexscrollbar.IndexIndicator,
EventType = {
/**
* Event triggered after select index by user
* @event select
* @member ns.widget.core.IndexScrollbar
*/
SELECT: "select"
},
POINTER_START = 'vmousedown',
POINTER_MOVE = 'vmousemove',
POINTER_END = 'vmouseup',
pointerIsPressed = false,
prototype = new BaseWidget();
IndexScrollbar.prototype = prototype;
utilsObject.merge(prototype, {
widgetName: "IndexScrollbar",
widgetClass: "ui-indexscrollbar",
_configure: function () {
/**
* All possible widget options
* @property {Object} options
* @property {string} [options.moreChar="*"] more character
* @property {string} [options.selectedClass="ui-state-selected"] disabled class name
* @property {string} [options.delimiter=","] delimiter in index
* @property {string|Array} [options.index=["A","B","C","D","E","F","G","H","I",
* "J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z","1"]]
* String with list of letters separate be delimiter or array of letters
* @property {boolean} [options.maxIndexLen=0]
* @property {boolean} [options.indexHeight=41]
* @property {boolean} [options.keepSelectEventDelay=50]
* @property {?boolean} [options.container=null]
* @property {?boolean} [options.supplementaryIndex=null]
* @property {number} [options.supplementaryIndexMargin=1]
* @member ns.widget.core.IndexScrollbar
*/
this.options = {
moreChar: "*",
indexScrollbarClass: "ui-indexscrollbar",
selectedClass: "ui-state-selected",
indicatorClass: "ui-indexscrollbar-indicator",
delimiter: ",",
index: [
"A", "B", "C", "D", "E", "F", "G", "H",
"I", "J", "K", "L", "M", "N", "O", "P", "Q",
"R", "S", "T", "U", "V", "W", "X", "Y", "Z", "1"
],
maxIndexLen: 0,
indexHeight: 41,
keepSelectEventDelay: 50,
container: null,
supplementaryIndex: null,
supplementaryIndexMargin: 1,
moreCharLineHeight: 9,
verticalCenter: true
};
},
/**
* This method builds widget.
* @method _build
* @protected
* @param {HTMLElement} element
* @return {HTMLElement}
* @member ns.widget.core.IndexScrollbar
*/
_build: function (element) {
return element;
},
/**
* This method inits widget.
* @method _init
* @protected
* @param {HTMLElement} element
* @return {HTMLElement}
* @member ns.widget.core.IndexScrollbar
*/
_init: function (element) {
var self = this,
options = self.options;
element.classList.add(options.indexScrollbarClass);
self._setIndex(element, options.index);
self._setMaxIndexLen(element, options.maxIndexLen);
self._setInitialLayout(); // This is needed for creating sub objects
self._createSubObjects();
self._updateLayout();
// Mark as extended
self._extended(true);
return element;
},
/**
* This method refreshes widget.
* @method _refresh
* @protected
* @return {HTMLElement}
* @member ns.widget.core.IndexScrollbar
*/
_refresh: function () {
if( this._isExtended() ) {
this._unbindEvent();
this.indicator.hide();
this._extended( false );
}
this._updateLayout();
this.indexBar1.options.index = this.options.index;
this.indexBar1.refresh();
this._bindEvents();
this._extended( true );
},
/**
* This method destroys widget.
* @method _destroy
* @protected
* @param {HTMLElement} element
* @return {HTMLElement}
* @member ns.widget.core.IndexScrollbar
*/
_destroy: function() {
var self = this;
if (self.isBound()) {
self._unbindEvent();
self._extended(false);
self._destroySubObjects();
self.indicator = null;
self.index = null;
self.eventHandlers = {};
}
},
/**
* This method creates indexBar1 and indicator in the indexScrollbar
* @method _createSubObjects
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_createSubObjects: function() {
var self = this,
options = self.options,
element = self.element;
// indexBar1
self.indexBar1 = new IndexBar( document.createElement("UL"), {
container: element,
offsetLeft: 0,
index: options.index,
verticalCenter: options.verticalCenter,
indexHeight: options.indexHeight,
maxIndexLen: options.maxIndexLen,
moreCharLineHeight: options.moreCharLineHeight
});
// indexBar2
if (typeof options.supplementaryIndex === "function") {
self.indexBar2 = new IndexBar( document.createElement("UL"), {
container: element,
offsetLeft: -element.clientWidth - options.supplementaryIndexMargin,
index: [], // empty index
indexHeight: options.indexHeight,
ulClass: "ui-indexscrollbar-supplementary"
});
self.indexBar2.hide();
}
// indicator
self.indicator = new IndexIndicator(document.createElement("DIV"), {
container: self._getContainer(),
referenceElement: self.element,
className: options.indicatorClass
});
},
/**
* This method destroys sub-elements: index bars and indicator.
* @method _destroySubObjects
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_destroySubObjects: function() {
var subObjs = {
iBar1: this.indexBar1,
iBar2: this.indexBar2,
indicator: this.indicator
},
subObj,
el,
i;
for(i in subObjs) {
subObj = subObjs[i];
if(subObj) {
el = subObj.element;
subObj.destroy();
el.parentNode.removeChild(el);
}
}
},
/**
* This method sets initial layout.
* @method _setInitialLayout
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_setInitialLayout: function () {
var indexScrollbar = this.element,
container = this._getContainer(),
containerPosition = window.getComputedStyle(container).position,
indexScrollbarStyle = indexScrollbar.style;
// Set the indexScrollbar's position, if needed
if (containerPosition !== "absolute" && containerPosition !== "relative") {
indexScrollbarStyle.top = container.offsetTop + "px";
indexScrollbarStyle.height = container.offsetHeight + "px";
}
},
/**
* This method calculates maximum index length.
* @method _setMaxIndexLen
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_setMaxIndexLen: function(element, value) {
var self = this,
options = self.options,
container = self._getContainer(),
containerHeight = container.offsetHeight;
if (value <= 0) {
value = Math.floor( containerHeight / options.indexHeight );
}
if (value > 0 && value%2 === 0) {
value -= 1; // Ensure odd number
}
options.maxIndexLen = value;
},
/**
* This method updates layout.
* @method _updateLayout
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_updateLayout: function() {
this._setInitialLayout();
this._draw();
this.touchAreaOffsetLeft = this.element.offsetLeft - 10;
},
/**
* This method draws additional sub-elements
* @method _draw
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_draw: function () {
this.indexBar1.show();
return this;
},
/**
* This method removes indicator.
* @method _removeIndicator
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_removeIndicator: function() {
var indicator = this.indicator,
parentElem = indicator.element.parentNode;
parentElem.removeChild(indicator.element);
indicator.destroy();
this.indicator = null;
},
/**
* This method returns the receiver of event by position.
* @method _getEventReceiverByPosition
* @param {number} posX The position relative to the left edge of the document.
* @return {?ns.widget.core.indexscrollbar.IndexBar} Receiver of event
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_getEventReceiverByPosition: function(posX) {
var windowWidth = window.innerWidth,
elementWidth = this.element.clientWidth,
receiver;
if( this.options.supplementaryIndex ) {
if( windowWidth - elementWidth <= posX && posX <= windowWidth) {
receiver = this.indexBar1;
} else {
receiver = this.indexBar2;
}
} else {
receiver = this.indexBar1;
}
return receiver;
},
/**
* This method updates indicator.
* It sets new value of indicator and triggers event "select".
* @method _updateIndicatorAndTriggerEvent
* @param {number} val The value of indicator
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_updateIndicatorAndTriggerEvent: function(val) {
this.indicator.setValue( val );
this.indicator.show();
if(this.selectEventTriggerTimeoutId) {
window.clearTimeout(this.selectEventTriggerTimeoutId);
}
this.selectEventTriggerTimeoutId = window.setTimeout(function() {
this.trigger(EventType.SELECT, {index: val});
this.selectEventTriggerTimeoutId = null;
}.bind(this), this.options.keepSelectEventDelay);
},
/**
* This method is executed on event "touchstart"
* @method _onTouchStartHandler
* @param {Event} event Event
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_onTouchStartHandler: function(event) {
pointerIsPressed = true;
var touches = event.touches || event._originalEvent && event._originalEvent.touches;
if (touches && (touches.length > 1)) {
event.preventDefault();
event.stopPropagation();
return;
}
var pos = this._getPositionFromEvent(event),
// At touchstart, only indexbar1 is shown.
iBar1 = this.indexBar1,
idx = iBar1.getIndexByPosition( pos.y ),
val = iBar1.getValueByIndex( idx );
iBar1.select( idx ); // highlight selected value
document.addEventListener(POINTER_MOVE, this.eventHandlers.touchMove);
document.addEventListener(POINTER_END, this.eventHandlers.touchEnd);
document.addEventListener("touchcancel", this.eventHandlers.touchEnd);
this._updateIndicatorAndTriggerEvent( val );
},
/**
* This method is executed on event "touchmove"
* @method _onTouchMoveHandler
* @param {Event} event Event
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_onTouchMoveHandler: function(event) {
var touches = event._originalEvent && event._originalEvent.touches;
if (touches && (touches.length > 1) || !pointerIsPressed) {
events.preventDefault(event);
events.stopPropagation(event);
return;
}
var pos = this._getPositionFromEvent( event ),
iBar1 = this.indexBar1,
iBar2 = this.indexBar2,
idx,
iBar,
val;
// Check event receiver: ibar1 or ibar2
iBar = this._getEventReceiverByPosition( pos.x );
if( iBar === iBar2 ) {
iBar2.options.index = this.options.supplementaryIndex(iBar1.getValueByIndex(iBar1.selectedIndex));
iBar2.refresh();
}
// get index and value from ibar1 or ibar2
idx = iBar.getIndexByPosition( pos.y );
val = iBar.getValueByIndex( idx );
if(iBar === iBar2) {
// Update val to make a concatenated string for indexIndicator
val = iBar1.getValueByIndex(iBar1.selectedIndex) + val;
} else if(iBar2 && !iBar2.isShown()) {
// iBar1 is selected.
// Set iBar2's paddingTop, only when the iBar2 isn't shown
iBar2.setPaddingTop(iBar1.getOffsetTopByIndex(iBar1.selectedIndex));
}
// update ibars
iBar.select(idx); // highlight selected value
iBar.show();
if( iBar1 === iBar && iBar2 ) {
iBar2.hide();
}
// update indicator
this._updateIndicatorAndTriggerEvent( val );
events.preventDefault(event);
events.stopPropagation(event);
},
/**
* This method is executed on event "touchend"
* @method _onTouchEndHandler
* @param {Event} event Event
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_onTouchEndHandler: function( event ) {
var self = this,
touches = event._originalEvent && event._originalEvent.touches;
if (touches && (touches.length === 0) ||
!touches) {
pointerIsPressed = false;
}
self.indicator.hide();
self.indexBar1.clearSelected();
if(self.indexBar2) {
self.indexBar2.clearSelected();
self.indexBar2.hide();
}
document.removeEventListener(POINTER_MOVE, self.eventHandlers.touchMove);
document.removeEventListener(POINTER_END, self.eventHandlers.touchEnd);
document.removeEventListener("touchcancel", self.eventHandlers.touchEnd);
},
/**
* This method binds events to widget.
* @method _bindEvents
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_bindEvents: function() {
this._bindResizeEvent();
this._bindEventToTriggerSelectEvent();
},
/**
* This method unbinds events to widget.
* @method _unbindEvent
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_unbindEvent: function() {
this._unbindResizeEvent();
this._unbindEventToTriggerSelectEvent();
},
/**
* This method binds event "resize".
* @method _bindResizeEvent
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_bindResizeEvent: function() {
this.eventHandlers.onresize = function(/* ev */) {
this.refresh();
}.bind(this);
window.addEventListener( "resize", this.eventHandlers.onresize );
},
/**
* This method unbinds event "resize".
* @method _bindResizeEvent
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_unbindResizeEvent: function() {
if ( this.eventHandlers.onresize ) {
window.removeEventListener( "resize", this.eventHandlers.onresize );
}
},
/**
* This method binds touch events.
* @method _bindEventToTriggerSelectEvent
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_bindEventToTriggerSelectEvent: function() {
var self = this;
self.eventHandlers.touchStart = self._onTouchStartHandler.bind(self);
self.eventHandlers.touchEnd = self._onTouchEndHandler.bind(self);
self.eventHandlers.touchMove = self._onTouchMoveHandler.bind(self);
self.element.addEventListener(POINTER_START, self.eventHandlers.touchStart);
},
/**
* This method unbinds touch events.
* @method _unbindEventToTriggerSelectEvent
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_unbindEventToTriggerSelectEvent: function() {
var self = this;
self.element.removeEventListener(POINTER_START, self.eventHandlers.touchStart);
},
/**
* This method sets or gets data from widget.
* @method _data
* @param {string|Object} key
* @param {*} val
* @return {*} Return value of data or widget's object
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_data: function (key, val) {
var el = this.element,
d = el.__data,
idx;
if(!d) {
d = el.__data = {};
}
if(typeof key === "object") {
// Support data collection
for(idx in key) {
this._data(idx, key[idx]);
}
return this;
} else {
if("undefined" === typeof val) { // Getter
return d[key];
} else { // Setter
d[key] = val;
return this;
}
}
},
/**
* This method checks if element is valid element of widget IndexScrollbar.
* @method _isValidElement
* @param {HTMLElement} el
* @return {boolean} True, if element is valid.
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_isValidElement: function (el) {
return el.classList.contains(this.widgetClass);
},
/**
* This method checks if widget is extended.
* @method _isExtended
* @return {boolean} True, if element is extended.
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_isExtended: function () {
return !!this._data("extended");
},
/**
* This method sets value of "extended" to widget.
* @method _extended
* @param {boolean} flag Value for extended
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_extended: function (flag) {
this._data("extended", flag);
return this;
},
/**
* This method gets indices prepared from parameter
* or index of widget.
* @method _setIndex
* @param {HTMLElement} element element
* @param {string} value Indices to prepared
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_setIndex: function (element, value) {
var options = this.options;
if (typeof value === "string") {
value = value.split(options.delimiter); // delimiter
}
options.index = value;
},
/**
* This method gets offset of element.
* @method _getOffset
* @param {HTMLElement} el Element
* @return {Object} Offset with "top" and "left" properties
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_getOffset: function( el ) {
var left=0, top=0 ;
do {
top += el.offsetTop;
left += el.offsetLeft;
el = el.offsetParent;
} while (el);
return {
top: top,
left: left
};
},
/**
* This method returns container of widget.
* @method _getContainer
* @return {HTMLElement} Container
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_getContainer: function() {
var container = this.options.container,
element = this.element,
parentElement = element.parentNode,
overflow;
if (!container) {
while (parentElement && parentElement != document.body) {
overflow = doms.getCSSProperty(parentElement, "overflow-y");
if (overflow === "scroll" || (overflow === "auto" && parentElement.scrollHeight > parentElement.clientHeight)) {
return parentElement;
}
parentElement = parentElement.parentNode;
}
container = element.parentNode;
}
return container || element.parentNode;
},
/**
* Returns position of event.
* @method _getPositionFromEvent
* @return {Object} Position of event with properties "x" and "y"
* @protected
* @member ns.widget.core.IndexScrollbar
*/
_getPositionFromEvent: function( ev ) {
return ev.type.search(/^touch/) !== -1 ?
{x: ev.touches[0].clientX, y: ev.touches[0].clientY} :
{x: ev.clientX, y: ev.clientY};
},
/**
* Adds event listener to element of widget.
* @method addEventListener
* @param {string} type Name of event
* @param {Function} listener Function to be executed
* @member ns.widget.core.IndexScrollbar
*/
addEventListener: function (type, listener) {
this.element.addEventListener(type, listener);
},
/**
* Removes event listener from element of widget.
* @method removeEventListener
* @param {string} type Name of event
* @param {Function} listener Function to be removed
* @member ns.widget.core.IndexScrollbar
*/
removeEventListener: function (type, listener) {
this.element.removeEventListener(type, listener);
}
});
// definition
ns.widget.core.IndexScrollbar = IndexScrollbar;
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, plusplus: true */
/*
*
* @class ns.widget.wearable.IndexScrollbar
* @extends ns.widget.core.IndexScrollbar
* @since 2.0
*/
(function (document, ns) {
"use strict";
var engine = ns.engine,
CoreIndexScrollbar = ns.widget.core.IndexScrollbar,
prototype = new CoreIndexScrollbar(),
IndexScrollbar = function () {
CoreIndexScrollbar.call(this);
};
// definition
IndexScrollbar.prototype = prototype;
ns.widget.wearable.IndexScrollbar = IndexScrollbar;
engine.defineWidget(
"IndexScrollbar",
".ui-indexscrollbar",
[],
IndexScrollbar,
"wearable"
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* # CircularIndexScrollbar UI Component
* Shows a circularindexscrollbar with indices, usually for the list.
*
* The circularindexscrollbar component shows on the screen a circularscrollbar with indices.
* The indices can be selected by moving the rotary.
* And it fires a select event when the index characters are selected.
*
* ## Manual constructor
* For manual creation of UI Component you can use constructor of component from **tau** namespace:
*
* @example
* var circularindexElement = document.getElementById('circularindex'),
* circularindexscrollbar = tau.widget.CircularIndexScrollbar(circularindexElement, {index: "A,B,C"});
*
* Constructor has one require parameter **element** which are base **HTMLElement** to create component.
* We recommend get this element by method *document.getElementById*. Second parameter is **options**
* and it is a object with options for component.
*
* To add an CircularIndexScrollbar component to the application, use the following code:
*
* @example
* <div id="foo" class="ui-circularindexscrollbar" data-index="A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,Z"></div>
* <script>
* (function() {
* var elem = document.getElementById("foo");
* tau.widget.CircularIndexScrollbar(elem);
* elem.addEventListener("select", function( event ) {
* var index = event.detail.index;
* console.log(index);
* });
* }());
* </script>
*
* The index value can be retrieved by accessing event.detail.index property.
*
* In the following example, the list scrolls to the position of the list item defined using
* the li-divider class, selected by the circularindexscrollbar:
*
* @example
* <div id="pageCircularIndexScrollbar" class="ui-page">
* <header class="ui-header">
* <h2 class="ui-title">CircularIndexScrollbar</h2>
* </header>
* <div id="circularindexscrollbar"class="ui-circularindexscrollbar" data-index="A,B,C,D,E"></div>
* <section class="ui-content">
* <ul class="ui-listview" id="list1">
* <li class="li-divider">A</li>
* <li>Anton</li>
* <li>Arabella</li>
* <li>Art</li>
* <li class="li-divider">B</li>
* <li>Barry</li>
* <li>Bibi</li>
* <li>Billy</li>
* <li>Bob</li>
* <li class="li-divider">D</li>
* <li>Daisy</li>
* <li>Derek</li>
* <li>Desmond</li>
* </ul>
* </section>
* <script>
* (function () {
* var page = document.getElementById("pageIndexScrollbar"),
circularindexscrollbar;
* page.addEventListener("pagecreate", function () {
* var elisb = document.getElementById("circularindexscrollbar"), // CircularIndexscrollbar element
* elList = document.getElementById("list1"), // List element
* elDividers = elList.getElementsByClassName("li-divider"), // List items (dividers)
* elScroller = elList.parentElement, // List's parent item
* dividers = {}, // Collection of list dividers
* indices = [], // List of index
* elDivider,
* i, idx;
*
* // For all list dividers
* for (i = 0; i < elDividers.length; i++) {
* // Add the list divider elements to the collection
* elDivider = elDividers[i];
* // li element having the li-divider class
* idx = elDivider.innerText;
* // Get a text (index value)
* dividers[idx] = elDivider;
* // Remember the element
*
* // Add the index to the index list
* indices.push(idx);
* }
*
* // Create CircularIndexScrollbar
* circularindexscrollbar = new tau.widget.CircularIndexScrollbar(elisb, {index: indices});
*
* // Bind the select callback
* elisb.addEventListener("select", function (ev) {
* var elDivider,
* idx = ev.detail.index;
* elDivider = dividers[idx];
* if (elDivider) {
* // Scroll to the li-divider element
* elScroller.scrollTop = elDivider.offsetTop - elScroller.offsetTop;
* }
* });
* });
* }());
* </script>
* </div>
*
* @author Junyoung Park <jy-.park@samsung.com>
* @author Hagun Kim <hagun.kim@samsung.com>
* @class ns.widget.wearable.CircularIndexScrollbar
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
utilsEvents = ns.event,
eventTrigger = utilsEvents.trigger,
prototype = new BaseWidget(),
CircularIndexScrollbar = function() {
this._phase = null;
this._tid = {
phaseOne: 0,
phaseThree: 0
};
this._detent = {
phaseOne: 0
};
this.options = {};
this._activeIndex = 0;
},
rotaryDirection = {
// right rotary direction
CW: "CW",
// left rotary direction
CCW: "CCW"
},
EventType = {
/**
* Event triggered after select index by user
* @event select
* @member ns.widget.wearable.CircularIndexScrollbar
*/
SELECT: "select"
},
classes = {
INDEXSCROLLBAR: "ui-circularindexscrollbar",
INDICATOR: "ui-circularindexscrollbar-indicator",
INDICATOR_TEXT: "ui-circularindexscrollbar-indicator-text",
SHOW: "ui-circularindexscrollbar-show"
};
CircularIndexScrollbar.prototype = prototype;
/**
* This method configure component.
* @method _configure
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._configure = function() {
/**
* All possible component options
* @property {Object} options
* @property {string} [options.delimiter=","] delimiter in index
* @property {string|Array} [options.index=["A","B","C","D","E","F","G","H","I","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z","1"]] indices list
* String with list of letters separate be delimiter or array of letters
* @property {number} [options.maxVisibleIndex=30] maximum length of visible indices
* @property {number} [options.duration=500] duration of show/hide animation time
* @member ns.widget.wearable.CircularIndexScrollbar
*/
this.options = {
delimiter: ",",
index: [
"A", "B", "C", "D", "E", "F", "G", "H",
"I", "J", "K", "L", "M", "N", "O", "P", "Q",
"R", "S", "T", "U", "V", "W", "X", "Y", "Z", "1"
]
};
};
/**
* This method build component.
* @method _build
* @protected
* @param {HTMLElement} element
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._build = function(element) {
var indicator,
indicatorText;
indicator = document.createElement("div");
indicator.classList.add(classes.INDICATOR);
indicatorText = document.createElement("span");
indicatorText.classList.add(classes.INDICATOR_TEXT);
indicator.appendChild(indicatorText);
element.appendChild(indicator);
element.classList.add(classes.INDEXSCROLLBAR);
return element;
};
/**
* This method inits component.
* @method _init
* @protected
* @param {HTMLElement} element
* @return {HTMLElement}
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._init = function(element) {
var self = this,
options = self.options;
self._phase = 1;
self._setIndices(options.index);
self._setValueByPosition(self._activeIndex, true);
return element;
};
/**
* This method set indices prepared from parameter
* or index of component.
* @method _setIndices
* @param {string} [value] Indices to prepared
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._setIndices = function(value) {
var self = this,
options = self.options;
if (value === null) {
ns.warn("CircularIndexScrollbar must have indices.");
options.index = null;
return;
}
if (typeof value === "string") {
value = value.split(options.delimiter); // delimiter
}
options.index = value;
};
/**
* This method select the index
* @method _setValueByPosition
* @protected
* @param {string} index index number
* @param {boolean} isFireEvent whether "select" event is fired or not
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._setValueByPosition = function(index, isFireEvent) {
var self = this,
indicatorText;
if (!self.options.index) {
return;
}
indicatorText = self.element.querySelector("." + classes.INDICATOR_TEXT);
self._activeIndex = index;
indicatorText.innerHTML = self.options.index[index];
if (isFireEvent) {
eventTrigger(self.element, EventType.SELECT, {index: self.options.index[index]});
}
};
/**
* This method select next index
* @method _nextIndex
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._nextIndex = function() {
var self = this,
activeIndex = self._activeIndex,
indexLen = self.options.index.length,
nextIndex;
if (activeIndex < indexLen -1 ) {
nextIndex = activeIndex + 1;
} else {
return;
}
self._setValueByPosition(nextIndex, true);
};
/**
* This method select previos index
* @method _prevIndex
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._prevIndex = function() {
var self = this,
activeIndex = self._activeIndex,
prevIndex;
if (activeIndex > 0) {
prevIndex = activeIndex - 1;
} else {
return;
}
self._setValueByPosition(prevIndex, true);
};
/**
* Get or Set index of the CircularIndexScrollbar
*
* Return current index or set the index
*
* @example
* <progress class="ui-circularindexscrollbar" id="circularindexscrollbar"></progress>
* <script>
* var circularindexElement = document.getElementById("circularindex"),
* circularIndexScrollbar = tau.widget.CircleProgressBar(circularindexElement),
* // return current index value
* value = circularIndexScrollbar.value();
* // sets the index value
* circularIndexScrollbar.value("C");
* </script>
* @method value
* return {string} In get mode return current index value
* @member ns.widget.wearable.CircularIndexScrollbar
*/
/**
* This method select the index
* @method _setValue
* @protected
* @param {string} value of index
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._setValue = function(value) {
var self = this,
index = self.options.index,
indexNumber;
if (index && (indexNumber = index.indexOf(value)) >= 0) {
self._setValueByPosition(indexNumber, false);
}
};
/**
* This method gets current index
* @method _getValue
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._getValue = function() {
var self = this,
index = self.options.index;
if (index) {
return index[self._activeIndex];
} else {
return null;
}
};
/**
* This method is a "rotarydetent" event handler
* @method _rotary
* @protected
* @param {Event} event Event
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._rotary = function(event) {
var self = this,
direction = event.detail.direction;
if (!self.options.index) {
return;
}
if (self._phase === 1) {
self._rotaryPhaseOne();
} else if (self._phase === 3) {
event.stopPropagation();
self._rotaryPhaseThree(direction);
}
};
/**
* This method is for phase 1 operation.
* @method _rotaryPhaseOne
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._rotaryPhaseOne = function() {
var self = this;
clearTimeout(self._tid.phaseOne);
self._tid.phaseOne = setTimeout(function(){
if (self._phase === 1) {
self._detent.phaseOne = 0;
}
}, 100);
if (self._detent.phaseOne > 3) {
self._phase = 3;
clearTimeout(self._tid.phaseOne);
self._detent.phaseOne = 0;
} else {
self._detent.phaseOne++;
}
};
/**
* This method is for phase 3 operation.
* @method _rotaryPhaseThree
* @protected
* @param {string} direction direction of rotarydetent event
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._rotaryPhaseThree = function(direction) {
var self = this;
clearTimeout(self._tid.phaseThree);
self._tid.phaseThree = setTimeout(function(){
self.element.classList.remove(classes.SHOW);
self._phase = 1;
}, 1000);
if(self._phase === 3) {
self.element.classList.add(classes.SHOW);
if (direction === rotaryDirection.CW) {
self._nextIndex();
} else {
self._prevIndex();
}
}
};
/**
* This method handles events
* @method handleEvent
* @public
* @param {Event} event Event
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype.handleEvent = function(event) {
var self = this;
switch (event.type) {
case "rotarydetent":
self._rotary(event);
break;
}
};
/**
* This method binds events to component.
* method _bindEvents
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._bindEvents = function() {
var self = this;
utilsEvents.on(document, "rotarydetent", self);
};
/**
* This method unbinds events to component.
* method _unbindEvents
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._unbindEvents = function() {
var self = this;
utilsEvents.off(document, "rotarydetent", self);
};
/**
* This method refreshes component.
* @method _refresh
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._refresh = function() {
var self = this,
options = self.options;
self._unbindEvents();
self._setIndices(options.index);
self._setValueByPosition(self._activeIndex, true);
self._bindEvents();
};
/**
* This method detroys component.
* @method _destroy
* @protected
* @member ns.widget.wearable.CircularIndexScrollbar
*/
prototype._destroy = function() {
var self = this;
self._unbindEvents();
};
// definition
ns.widget.wearable.CircularIndexScrollbar = CircularIndexScrollbar;
engine.defineWidget(
"CircularIndexScrollbar",
".ui-circularindexscrollbar",
[],
CircularIndexScrollbar,
"wearable"
);
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Progress Widget
* Shows a control that indicates the progress percentage of an on-going operation.
*
* The progress widget shows a control that indicates the progress percentage of an on-going operation. This widget can be scaled to fit inside a parent container.
*
* ## Default selectors
*
* This widget provide three style progress.
*
* ### Simple progress bar
* If you don't implement any class, you can show default progress style
* To add a progress widget to the application, use the following code:
*
* @example
* <progress max="100" value="90"></progress>
*
* ### Infinite progress bar
* If you implement class (*ui-progress-indeterminate*), you can show image looks like infinite move.
*
* To add a progress widget to the application, use the following code:
* @example
* <progress class="ui-progress-indeterminate" max="100" value="100"></progress>
*
* ### Progress bar with additional information
* If you implement div tag that can choose two classes (*ui-progress-proportion* or *ui-progress-ratio*) at progress tag same level, you can show two information (proportion information is located left below and ratio information is located right below)
*
* To add a progress widget to the application, use the following code:
*
* @example
* <progress max="100" value="50"></progress>
* <div class="ui-progress-proportion">00/20</div>
* <div class="ui-progress-ratio">50%</div>
*
* ## JavaScript API
*
* Progress widget hasn't JavaScript API.
*
* @class ns.widget.wearable.Progress
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
Progress = function () {
return this;
},
prototype = new BaseWidget();
Progress.events = {};
/**
* Build Progress
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.wearable.Progress
*/
prototype._build = function (element) {
return element;
};
prototype._init = function (element) {
return element;
};
prototype._bindEvents = function (element) {
return element;
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.wearable.Progress
*/
prototype._refresh = function () {
return null;
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.wearable.Progress
*/
prototype._destroy = function () {
return null;
};
Progress.prototype = prototype;
ns.widget.wearable.Progress = Progress;
engine.defineWidget(
"Progress",
"progress",
[],
Progress,
"wearable"
);
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Processing Widget
* Shows that an operation is in progress.
*
* The processing widget shows that an operation is in progress.
*
* ## Default selectors
*
* To add a processing widget to the application, use the following code:
*
* @example
* <div class="ui-processing"></div>
* <div class="ui-processing-text">
* Description about progress
* </div>
*
* ## JavaScript API
*
* Processing widget hasn't JavaScript API.
*
* @class ns.widget.wearable.Progressing
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
Progressing = function () {
return this;
},
prototype = new BaseWidget();
Progressing.events = {};
/**
* Build Progressing
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.wearable.Progressing
*/
prototype._build = function (element) {
return element;
};
prototype._init = function (element) {
return element;
};
prototype._bindEvents = function (element) {
return element;
};
/**
* Refresh structure
* @method _refresh
* @protected
* @member ns.widget.wearable.Progressing
*/
prototype._refresh = function () {
return null;
};
/**
* Destroy widget
* @method _destroy
* @protected
* @member ns.widget.wearable.Progressing
*/
prototype._destroy = function () {
return null;
};
Progressing.prototype = prototype;
ns.widget.wearable.Progressing = Progressing;
engine.defineWidget(
"Progressing",
".ui-progress",
[],
Progressing,
"wearable"
);
}(window.document, ns));
/*global window, define */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true */
/**
* # Toggle Switch Widget
* Shows a 2-state switch.
*
* The toggle switch widget shows a 2-state switch on the screen.
*
* ## Default selectors
*
* To add a toggle switch widget to the application, use the following code:
*
* @example
* <div class="ui-switch">
* <div class="ui-switch-text">
* Toggle Switch
* </div>
* <label class="ui-toggleswitch">
* <input type="checkbox" class="ui-switch-input">
* <div class="ui-switch-activation">
* <div class="ui-switch-inneroffset">
* <div class="ui-switch-handler"></div>
* </div>
* </div>
* </label>
* </div>
*
* ## JavaScript API
*
* ToggleSwitch widget hasn't JavaScript API.
*
* @class ns.widget.wearable.ToggleSwitch
* @extends ns.widget.BaseWidget
*/
(function (document, ns) {
"use strict";
var BaseWidget = ns.widget.BaseWidget,
engine = ns.engine,
ToggleSwitch = function () {
/**
* Options for widget
* @property {Object} options
* @property {?string} [options.text=null] Shown text
* @member ns.widget.wearable.ToggleSwitch
*/
this.options = {
text: null
};
},
events = {},
classesPrefix = "ui-switch",
classes = {
handler: classesPrefix + "-handler",
inneroffset: classesPrefix + "-inneroffset",
activation: classesPrefix + "-activation",
input: classesPrefix + "-input",
text: classesPrefix + "-text"
},
prototype = new BaseWidget();
function getClass(name) {
return classes[name];
}
function addClass(element, classId) {
element.classList.add(getClass(classId));
}
function createElement(name) {
return document.createElement(name);
}
/**
* Dictionary for ToggleSwitch related events.
* For ToggleSwitch, it is an empty object.
* @property {Object} events
* @member ns.widget.wearable.ToggleSwitch
* @static
*/
ToggleSwitch.events = events;
/**
* Dictionary for ToggleSwitch related css class names
* @property {Object} classes
* @member ns.widget.wearable.ToggleSwitch
* @static
* @readonly
*/
ToggleSwitch.classes = classes;
/**
* Build ToggleSwitch
* @method _build
* @param {HTMLElement} element
* @return {HTMLElement}
* @protected
* @member ns.widget.wearable.ToggleSwitch
*/
prototype._build = function (element) {
var options = this.options,
text = options.text,
divText,
label = createElement("label"),
input = createElement("input"),
divActivation = createElement("div"),
divInneroffset = createElement("div"),
divHandler = createElement("div");
if (text) {
divText = createElement("div");
addClass(divText, "text");
divText.innerHTML = text;
element.appendChild(divText);
}
addClass(divHandler, "handler");
divInneroffset.appendChild(divHandler);
addClass(divInneroffset, "inneroffset");
divActivation.appendChild(divInneroffset);
addClass(divActivation, "activation");
label.classList.add("ui-toggleswitch");
input.type = "checkbox";
addClass(input, "input");
label.appendChild(input);
label.appendChild(divActivation);
element.appendChild(label);
return element;
};
ToggleSwitch.prototype = prototype;
ns.widget.wearable.ToggleSwitch = ToggleSwitch;
engine.defineWidget(
"ToggleSwitch",
".ui-switch",
[],
ToggleSwitch,
"wearable"
);
}(window.document, ns));
/*global window, define, ns */
/*
* Copyright (c) 2015 Samsung Electronics Co., Ltd
*
* Licensed under the Flora License, Version 1.1 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://floralicense.org/license/
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*jslint nomen: true, white: true, plusplus: true*/
(function (document, ns) {
"use strict";
/**
* @property {Object} Widget Alias for {@link ns.widget.BaseWidget}
* @member ns.widget.core.VirtualListview
* @private
* @static
*/
var BaseWidget = ns.widget.BaseWidget,
// Constants definition
/**
* Defines index of scroll `{@link ns.widget.core.VirtualListview#_scroll}.direction`
* @property {number} SCROLL_UP
* to retrive if user is scrolling up
* @private
* @static
* @member ns.widget.core.VirtualListview
*/
SCROLL_UP = 0,
/**
* Defines index of scroll `{@link ns.widget.core.VirtualListview#_scroll}.direction`
* @property {number} SCROLL_RIGHT
* to retrive if user is scrolling right
* @private
* @static
* @member ns.widget.core.VirtualListview
*/
SCROLL_RIGHT = 1,
/**
* Defines index of scroll {@link ns.widget.core.VirtualListview#_scroll}
* @property {number} SCROLL_DOWN
* to retrive if user is scrolling down
* @private
* @static
* @member ns.widget.core.VirtualListview
*/
SCROLL_DOWN = 2,
/**
* Defines index of scroll {@link ns.widget.core.VirtualListview#_scroll}
* @property {number} SCROLL_LEFT
* to retrive if user is scrolling left
* @private
* @static
* @member ns.widget.core.VirtualListview
*/
SCROLL_LEFT = 3,
/**
* Defines ver
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment