2020-10-02 15:10:00 +00:00
|
|
|
import {urlState} from 'app/client/models/gristUrlState';
|
2021-08-18 17:49:34 +00:00
|
|
|
import {getTheme} from 'app/client/ui/CustomThemes';
|
2020-10-02 15:10:00 +00:00
|
|
|
import {cssLeftPane} from 'app/client/ui/PagePanels';
|
|
|
|
import {colors, testId, vars} from 'app/client/ui2018/cssVars';
|
|
|
|
import * as version from 'app/common/version';
|
2021-08-18 17:49:34 +00:00
|
|
|
import {BindableValue, Disposable, dom, styled} from "grainjs";
|
2021-11-05 14:47:17 +00:00
|
|
|
import {menu, menuDivider, menuItem, menuItemLink, menuSubHeader} from 'app/client/ui2018/menus';
|
|
|
|
import {Organization, SUPPORT_EMAIL} from 'app/common/UserAPI';
|
2021-08-18 17:49:34 +00:00
|
|
|
import {AppModel} from 'app/client/models/AppModel';
|
|
|
|
import {icon} from 'app/client/ui2018/icons';
|
2021-11-05 14:47:17 +00:00
|
|
|
import {DocPageModel} from 'app/client/models/DocPageModel';
|
|
|
|
import * as roles from 'app/common/roles';
|
|
|
|
import {loadUserManager} from 'app/client/lib/imports';
|
|
|
|
import {buildSiteSwitcher} from 'app/client/ui/SiteSwitcher';
|
2021-08-18 17:49:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
export class AppHeader extends Disposable {
|
2021-11-05 14:47:17 +00:00
|
|
|
constructor(private _orgName: BindableValue<string>, private _appModel: AppModel,
|
|
|
|
private _docPageModel?: DocPageModel) {
|
2021-08-18 17:49:34 +00:00
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
public buildDom() {
|
|
|
|
const theme = getTheme(this._appModel.topAppModel.productFlavor);
|
|
|
|
|
2021-11-05 14:47:17 +00:00
|
|
|
const user = this._appModel.currentValidUser;
|
|
|
|
const orgs = this._appModel.topAppModel.orgs;
|
|
|
|
const currentOrg = this._appModel.currentOrg;
|
|
|
|
const isTeamSite = Boolean(currentOrg && !currentOrg.owner);
|
|
|
|
const isBillingManager = Boolean(currentOrg && currentOrg.billingAccount &&
|
|
|
|
(currentOrg.billingAccount.isManager || user?.email === SUPPORT_EMAIL));
|
|
|
|
|
|
|
|
// Opens the user-manager for the org.
|
|
|
|
const manageUsers = async (org: Organization) => {
|
|
|
|
const api = this._appModel.api;
|
|
|
|
(await loadUserManager()).showUserManagerModal(api, {
|
|
|
|
permissionData: api.getOrgAccess(org.id),
|
|
|
|
activeEmail: user ? user.email : null,
|
|
|
|
resourceType: 'organization',
|
|
|
|
resourceId: org.id,
|
|
|
|
resource: org,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2021-08-18 17:49:34 +00:00
|
|
|
return cssAppHeader(
|
|
|
|
cssAppHeader.cls('-widelogo', theme.wideLogo || false),
|
|
|
|
// Show version when hovering over the application icon.
|
|
|
|
cssAppLogo(
|
|
|
|
{title: `Ver ${version.version} (${version.gitcommit})`},
|
|
|
|
urlState().setLinkUrl({}),
|
|
|
|
testId('dm-logo')
|
|
|
|
),
|
|
|
|
cssOrg(
|
|
|
|
cssOrgName(dom.text(this._orgName)),
|
|
|
|
this._orgName && cssDropdownIcon('Dropdown'),
|
2021-11-05 14:47:17 +00:00
|
|
|
menu(() => [
|
|
|
|
menuSubHeader(`${isTeamSite ? 'Team' : 'Personal'} Site`, testId('orgmenu-title')),
|
|
|
|
menuItemLink(urlState().setLinkUrl({}), 'Home Page', testId('orgmenu-home-page')),
|
2021-08-18 17:49:34 +00:00
|
|
|
|
2021-11-05 14:47:17 +00:00
|
|
|
// Show 'Organization Settings' when on a home page of a valid org.
|
|
|
|
(!this._docPageModel && currentOrg && !currentOrg.owner ?
|
|
|
|
menuItem(() => manageUsers(currentOrg), 'Manage Team', testId('orgmenu-manage-team'),
|
|
|
|
dom.cls('disabled', !roles.canEditAccess(currentOrg.access))) :
|
|
|
|
// Don't show on doc pages, or for personal orgs.
|
|
|
|
null),
|
|
|
|
|
|
|
|
// Show link to billing pages.
|
|
|
|
currentOrg && !currentOrg.owner ?
|
|
|
|
// For links, disabling with just a class is hard; easier to just not make it a link.
|
|
|
|
// TODO weasel menus should support disabling menuItemLink.
|
|
|
|
(isBillingManager ?
|
|
|
|
menuItemLink(urlState().setLinkUrl({billing: 'billing'}), 'Billing Account') :
|
|
|
|
menuItem(() => null, 'Billing Account', dom.cls('disabled', true), testId('orgmenu-billing'))
|
|
|
|
) :
|
|
|
|
null,
|
2021-08-18 17:49:34 +00:00
|
|
|
|
2021-11-05 14:47:17 +00:00
|
|
|
dom.maybe((use) => use(orgs).length > 0, () => [
|
|
|
|
menuDivider(),
|
|
|
|
buildSiteSwitcher(this._appModel),
|
|
|
|
]),
|
|
|
|
], { placement: 'bottom-start' }),
|
|
|
|
testId('dm-org'),
|
2021-08-18 17:49:34 +00:00
|
|
|
),
|
2021-11-05 14:47:17 +00:00
|
|
|
);
|
2021-08-18 17:49:34 +00:00
|
|
|
}
|
2020-10-02 15:10:00 +00:00
|
|
|
}
|
|
|
|
|
2021-08-18 17:49:34 +00:00
|
|
|
const cssAppHeader = styled('div', `
|
2020-10-02 15:10:00 +00:00
|
|
|
display: flex;
|
|
|
|
width: 100%;
|
|
|
|
height: 100%;
|
|
|
|
align-items: center;
|
|
|
|
&, &:hover, &:focus {
|
|
|
|
text-decoration: none;
|
|
|
|
outline: none;
|
|
|
|
color: ${colors.dark};
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
2021-08-18 17:49:34 +00:00
|
|
|
const cssAppLogo = styled('a', `
|
2020-10-02 15:10:00 +00:00
|
|
|
flex: none;
|
|
|
|
height: 48px;
|
|
|
|
width: 48px;
|
|
|
|
background-image: var(--icon-GristLogo);
|
|
|
|
background-size: 22px 22px;
|
|
|
|
background-repeat: no-repeat;
|
|
|
|
background-position: center;
|
|
|
|
background-color: ${vars.logoBg};
|
|
|
|
.${cssAppHeader.className}-widelogo & {
|
|
|
|
width: 100%;
|
|
|
|
background-size: contain;
|
|
|
|
background-origin: content-box;
|
|
|
|
padding: 8px;
|
|
|
|
}
|
|
|
|
.${cssLeftPane.className}-open .${cssAppHeader.className}-widelogo & {
|
|
|
|
background-image: var(--icon-GristWideLogo, var(--icon-GristLogo));
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
|
2021-08-18 17:49:34 +00:00
|
|
|
const cssDropdownIcon = styled(icon, `
|
|
|
|
flex-shrink: 0;
|
|
|
|
margin-right: 8px;
|
|
|
|
`);
|
|
|
|
|
|
|
|
const cssOrg = styled('div', `
|
|
|
|
display: flex;
|
|
|
|
flex-grow: 1;
|
|
|
|
align-items: center;
|
|
|
|
max-width: calc(100% - 48px);
|
|
|
|
cursor: pointer;
|
|
|
|
height: 100%;
|
|
|
|
`);
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
const cssOrgName = styled('div', `
|
2021-08-18 17:49:34 +00:00
|
|
|
padding-left: 16px;
|
|
|
|
padding-right: 8px;
|
2020-10-02 15:10:00 +00:00
|
|
|
white-space: nowrap;
|
|
|
|
overflow: hidden;
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
.${cssAppHeader.className}-widelogo & {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
`);
|