/* * Copyright (c) 2016 Washington State Department of Transportation * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/> * */ package gov.wa.wsdot.mobile.client.activities.mountainpasses; import com.google.gwt.aria.client.CheckedValue; import com.google.gwt.aria.client.Roles; import com.google.gwt.aria.client.SelectedValue; import com.google.gwt.core.client.GWT; import com.google.gwt.event.logical.shared.SelectionEvent; import com.google.gwt.uibinder.client.UiBinder; import com.google.gwt.uibinder.client.UiField; import com.google.gwt.uibinder.client.UiHandler; import com.google.gwt.user.client.ui.Composite; import com.google.gwt.user.client.ui.HTML; import com.google.gwt.user.client.ui.Widget; import com.googlecode.mgwt.dom.client.event.tap.TapEvent; import com.googlecode.mgwt.ui.client.MGWT; import com.googlecode.mgwt.ui.client.widget.button.image.NotimportantImageButton; import com.googlecode.mgwt.ui.client.widget.header.HeaderTitle; import com.googlecode.mgwt.ui.client.widget.image.ImageHolder; import com.googlecode.mgwt.ui.client.widget.list.celllist.CellList; import com.googlecode.mgwt.ui.client.widget.list.celllist.CellSelectedEvent; import com.googlecode.mgwt.ui.client.widget.panel.flex.FlexSpacer; import com.googlecode.mgwt.ui.client.widget.panel.flex.RootFlexPanel; import com.googlecode.mgwt.ui.client.widget.panel.scroll.ScrollPanel; import com.googlecode.mgwt.ui.client.widget.tabbar.TabPanel; import gov.wa.wsdot.mobile.client.activities.camera.CameraCell; import gov.wa.wsdot.mobile.client.widget.CellDetailsWithIcon; import gov.wa.wsdot.mobile.client.widget.button.image.BackImageButton; import gov.wa.wsdot.mobile.client.widget.tabbar.CameraTabBarButton; import gov.wa.wsdot.mobile.client.widget.tabbar.ForecastTabBarButton; import gov.wa.wsdot.mobile.client.widget.tabbar.ReportTabBarButton; import gov.wa.wsdot.mobile.shared.CameraItem; import gov.wa.wsdot.mobile.shared.ForecastItem; import java.util.List; public class MountainPassDetailsViewGwtImpl extends Composite implements MountainPassDetailsView { /** * The UiBinder interface. */ interface MountainPassDetailsViewGwtImplUiBinder extends UiBinder<Widget, MountainPassDetailsViewGwtImpl> { } /** * The UiBinder used to generate the view. */ private static MountainPassDetailsViewGwtImplUiBinder uiBinder = GWT .create(MountainPassDetailsViewGwtImplUiBinder.class); @UiField HeaderTitle heading; @UiField RootFlexPanel report; @UiField RootFlexPanel cameras; @UiField RootFlexPanel forecast; @UiField BackImageButton backButton; @UiField FlexSpacer leftFlexSpacer; @UiField(provided = true) NotimportantImageButton starButton; @UiField HTML title; @UiField HTML dateUpdated; @UiField HTML weatherCondition; @UiField HTML temperatureInFahrenheit; @UiField HTML elevationInFeet; @UiField HTML roadCondition; @UiField HTML restrictionOneTravelDirection; @UiField HTML restrictionOneText; @UiField HTML restrictionTwoTravelDirection; @UiField HTML restrictionTwoText; @UiField(provided = true) CellList<CameraItem> cameraCellList; @UiField(provided = true) CellList<ForecastItem> forecastCellList; @UiField TabPanel tabPanel; @UiField ReportTabBarButton reportTab; @UiField CameraTabBarButton camerasTab; @UiField ForecastTabBarButton forecastTab; @UiField ScrollPanel reportScrollPanel; @UiField static ScrollPanel cameraScrollPanel; @UiField ScrollPanel forecastScrollPanel; private Presenter presenter; public MountainPassDetailsViewGwtImpl() { starButton = new NotimportantImageButton(); handleOnLoad(); cameraCellList = new CellList<CameraItem>(new CameraCell<CameraItem>() { @Override public String getUrl(CameraItem model) { return model.getImageUrl(); } @Override public boolean canBeSelected(CameraItem model) { return true; } }); forecastCellList = new CellList<ForecastItem>(new CellDetailsWithIcon<ForecastItem>() { @Override public String getDisplayString(ForecastItem model) { return model.getDay(); } @Override public String getDisplayImage(ForecastItem model) { return model.getWeatherIcon(); } @Override public String getDisplayDescription(ForecastItem model) { return model.getForecastText(); } @Override public String getDisplayLastUpdated(ForecastItem model) { return ""; } @Override public boolean canBeSelected(ForecastItem model) { return false; } }); initWidget(uiBinder.createAndBindUi(this)); accessibilityPrepare(); if (MGWT.getOsDetection().isAndroid()) { leftFlexSpacer.setVisible(false); reportScrollPanel.setBounce(false); cameraScrollPanel.setBounce(false); forecastScrollPanel.setBounce(false); } } /** * ScrollPanel doesn't allow scrolling to the bottom if it contains a CellList with images. * * See: https://code.google.com/p/mgwt/issues/detail?id=276 * * ScrollPanel.refresh() must be explicitly called after the images are loaded. * Since the onload event of images is not bubbling up, the LoadHandler can't be attached * to the CellList. Instead, the onload event needs to be captured at the <img>, and directly * trigger the ScrollPanel.refresh() from there. */ private native void handleOnLoad() /*-{ $wnd.refreshPanel = @gov.wa.wsdot.mobile.client.activities.mountainpasses.MountainPassDetailsViewGwtImpl::refreshPanel(); }-*/; public static void refreshPanel() { cameraScrollPanel.refresh(); } @UiHandler("tabPanel") protected void onTabSelected(SelectionEvent<Integer> event) { if (presenter != null) { int index = event.getSelectedItem(); presenter.onTabSelected(index); } } @UiHandler("reportTab") protected void onReportTabPressed(TapEvent event) { accessibilityShowReport(); } @UiHandler("camerasTab") protected void onCamerasTabPressed(TapEvent event) { accessibilityShowCameras(); } @UiHandler("forecastTab") protected void onForecastTabPressed(TapEvent event) { accessibilityShowForcast(); } @UiHandler("backButton") protected void onBackButtonPressed(TapEvent event) { if (presenter != null) { presenter.onBackButtonPressed(); } } @UiHandler("starButton") protected void onStarButtonPressed(TapEvent event) { if (presenter != null) { presenter.onStarButtonPressed(); } } @UiHandler("cameraCellList") protected void onCameraCellSelected(CellSelectedEvent event) { if (presenter != null) { int index = event.getIndex(); presenter.onCameraSelected(index); } } @Override public void setPresenter(Presenter presenter) { this.presenter = presenter; } @Override public void setTitle(String title) { this.title.setHTML(title); } @Override public void setDateUpdated(String dateUpdated) { this.dateUpdated.setHTML(dateUpdated); } @Override public void setWeatherCondition(String weatherCondition) { this.weatherCondition.setHTML(weatherCondition); } @Override public void setTemperatureInFahrenheit(String temperatureInFahrenheit) { this.temperatureInFahrenheit.setHTML(temperatureInFahrenheit); } @Override public void setElevationInFeet(String elevationInFeet) { this.elevationInFeet.setHTML(elevationInFeet + " ft"); } @Override public void setRoadCondition(String roadCondition) { this.roadCondition.setHTML(roadCondition); } @Override public void setRestrictionOneTravelDirection( String restrictionOneTravelDirection) { this.restrictionOneTravelDirection.setHTML("Restrictions " + restrictionOneTravelDirection + ":"); } @Override public void setRestrictionOneText(String restrictionOneText) { this.restrictionOneText.setHTML(restrictionOneText); } @Override public void setRestrictionTwoTravelDirection( String restrictionTwoTravelDirection) { this.restrictionTwoTravelDirection.setHTML("Restrictions " + restrictionTwoTravelDirection + ":"); } @Override public void setRestrictionTwoText(String restrictionTwoText) { this.restrictionTwoText.setHTML(restrictionTwoText); } @Override public void removeTab(int tabIndex) { this.tabPanel.tabBar.remove(tabIndex); this.tabPanel.tabContainer.container.remove(tabIndex); this.tabPanel.tabContainer.refresh(); } @Override public void renderCamera(List<CameraItem> createCameraList) { cameraCellList.render(createCameraList); } @Override public void setCameraSelected(int lastIndex, boolean b) { cameraCellList.setSelectedIndex(lastIndex, b); } @Override public void renderForecast(List<ForecastItem> createForecastList) { forecastCellList.render(createForecastList); } @Override public void toggleStarButton(boolean isStarred) { if (isStarred) { starButton.setIcon(ImageHolder.get().important()); Roles.getCheckboxRole().setAriaCheckedState(starButton.getElement(), CheckedValue.TRUE); } else { starButton.setIcon(ImageHolder.get().notImportant()); Roles.getCheckboxRole().setAriaCheckedState(starButton.getElement(), CheckedValue.FALSE); } } @Override public void refreshReport() { tabPanel.tabContainer.refresh(); reportScrollPanel.refresh(); } @Override public void refreshCameras() { tabPanel.tabContainer.refresh(); cameraScrollPanel.refresh(); } @Override public void refreshForecast() { tabPanel.tabContainer.refresh(); forecastScrollPanel.refresh(); } private void accessibilityShowReport(){ Roles.getMainRole().setAriaHiddenState(report.getElement(), false); Roles.getMainRole().setAriaHiddenState(cameras.getElement(), true); Roles.getMainRole().setAriaHiddenState(forecast.getElement(), true); Roles.getTabRole().setAriaSelectedState(reportTab.getElement(), SelectedValue.TRUE); Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.FALSE); Roles.getTabRole().setAriaSelectedState(forecastTab.getElement(), SelectedValue.FALSE); } private void accessibilityShowCameras(){ Roles.getMainRole().setAriaHiddenState(report.getElement(), true); Roles.getMainRole().setAriaHiddenState(cameras.getElement(), false); Roles.getMainRole().setAriaHiddenState(forecast.getElement(), true); Roles.getTabRole().setAriaSelectedState(reportTab.getElement(), SelectedValue.FALSE); Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.TRUE); Roles.getTabRole().setAriaSelectedState(forecastTab.getElement(), SelectedValue.FALSE); } private void accessibilityShowForcast(){ Roles.getMainRole().setAriaHiddenState(report.getElement(), true); Roles.getMainRole().setAriaHiddenState(cameras.getElement(), true); Roles.getMainRole().setAriaHiddenState(forecast.getElement(), false); Roles.getTabRole().setAriaSelectedState(reportTab.getElement(), SelectedValue.FALSE); Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.FALSE); Roles.getTabRole().setAriaSelectedState(forecastTab.getElement(), SelectedValue.TRUE); } private void accessibilityPrepare(){ // Add ARIA roles for accessibility Roles.getButtonRole().set(backButton.getElement()); Roles.getButtonRole().setAriaLabelProperty(backButton.getElement(), "back"); Roles.getHeadingRole().set(heading.getElement()); Roles.getCheckboxRole().set(starButton.getElement()); Roles.getCheckboxRole().setAriaLabelProperty(starButton.getElement(), "favorite"); Roles.getTabRole().set(reportTab.getElement()); Roles.getTabRole().setAriaSelectedState(reportTab.getElement(), SelectedValue.TRUE); Roles.getTabRole().set(camerasTab.getElement()); Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.FALSE); Roles.getTabRole().set(forecastTab.getElement()); Roles.getTabRole().setAriaSelectedState(forecastTab.getElement(), SelectedValue.FALSE); accessibilityShowReport(); } }