signal-desktop/ts/window.d.ts

573 lines
21 KiB
TypeScript
Raw Normal View History

2022-01-05 17:59:59 +00:00
// Copyright 2020-2022 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
// Captures the globals put in place by preload.js, background.js and others
import { Store } from 'redux';
import * as Backbone from 'backbone';
import * as Underscore from 'underscore';
import moment from 'moment';
import PQueue from 'p-queue/dist';
2021-09-22 20:59:54 +00:00
import { Ref } from 'react';
import { imageToBlurHash } from './util/imageToBlurHash';
import * as Util from './util';
import {
ConversationModelCollectionType,
MessageModelCollectionType,
} from './model-types.d';
2021-07-14 23:39:52 +00:00
import { TextSecureType } from './textsecure.d';
import { Storage } from './textsecure/Storage';
import {
ChallengeHandler,
IPCRequest as IPCChallengeRequest,
} from './challenge';
import { WebAPIConnectType } from './textsecure/WebAPI';
2020-09-04 18:27:12 +00:00
import { CallingClass } from './services/calling';
2020-10-06 17:06:34 +00:00
import * as Groups from './groups';
import * as Crypto from './Crypto';
import * as Curve from './Curve';
import * as RemoteConfig from './RemoteConfig';
import * as OS from './OS';
import { getEnvironment } from './environment';
2022-05-11 22:58:14 +00:00
import { LocalizerType, ThemeType } from './types/Util';
import type { Receipt } from './types/Receipt';
import { ConversationController } from './ConversationController';
import { ReduxActions } from './state/types';
2020-10-30 17:52:21 +00:00
import { createStore } from './state/createStore';
2021-06-14 19:01:00 +00:00
import { createApp } from './state/roots/createApp';
2021-05-28 16:15:17 +00:00
import { createChatColorPicker } from './state/roots/createChatColorPicker';
import { createConversationDetails } from './state/roots/createConversationDetails';
2021-04-27 22:35:35 +00:00
import { createForwardMessageModal } from './state/roots/createForwardMessageModal';
import { createGroupLinkManagement } from './state/roots/createGroupLinkManagement';
import { createGroupV1MigrationModal } from './state/roots/createGroupV1MigrationModal';
import { createGroupV2JoinModal } from './state/roots/createGroupV2JoinModal';
import { createGroupV2Permissions } from './state/roots/createGroupV2Permissions';
2020-10-30 17:52:21 +00:00
import { createLeftPane } from './state/roots/createLeftPane';
import { createMessageDetail } from './state/roots/createMessageDetail';
2021-08-05 12:35:33 +00:00
import { createConversationNotificationsSettings } from './state/roots/createConversationNotificationsSettings';
import { createPendingInvites } from './state/roots/createPendingInvites';
2020-10-30 17:52:21 +00:00
import { createSafetyNumberViewer } from './state/roots/createSafetyNumberViewer';
import { createShortcutGuideModal } from './state/roots/createShortcutGuideModal';
import { createStickerManager } from './state/roots/createStickerManager';
import { createStickerPreviewModal } from './state/roots/createStickerPreviewModal';
2021-06-14 19:01:00 +00:00
import * as appDuck from './state/ducks/app';
2020-10-30 17:52:21 +00:00
import * as callingDuck from './state/ducks/calling';
import * as conversationsDuck from './state/ducks/conversations';
import * as emojisDuck from './state/ducks/emojis';
import * as expirationDuck from './state/ducks/expiration';
import * as itemsDuck from './state/ducks/items';
2021-04-27 22:35:35 +00:00
import * as linkPreviewsDuck from './state/ducks/linkPreviews';
2020-10-30 17:52:21 +00:00
import * as networkDuck from './state/ducks/network';
import * as updatesDuck from './state/ducks/updates';
import * as userDuck from './state/ducks/user';
import * as searchDuck from './state/ducks/search';
import * as stickersDuck from './state/ducks/stickers';
import * as conversationsSelectors from './state/selectors/conversations';
import * as searchSelectors from './state/selectors/search';
2020-09-09 02:25:05 +00:00
import AccountManager from './textsecure/AccountManager';
import { ContactWithHydratedAvatar } from './textsecure/SendMessage';
import Data from './sql/Client';
import { PhoneNumberFormat } from 'google-libphonenumber';
import { MessageModel } from './models/messages';
import { ConversationModel } from './models/conversations';
import { BatcherType } from './util/batcher';
import { AttachmentList } from './components/conversation/AttachmentList';
2021-05-28 16:15:17 +00:00
import { ChatColorPicker } from './components/ChatColorPicker';
import { ConfirmationDialog } from './components/ConfirmationDialog';
import { ContactDetail } from './components/conversation/ContactDetail';
import { ContactModal } from './components/conversation/ContactModal';
2020-10-06 17:06:34 +00:00
import { ErrorModal } from './components/ErrorModal';
import { Lightbox } from './components/Lightbox';
import { MediaGallery } from './components/conversation/media-gallery/MediaGallery';
import { MessageDetail } from './components/conversation/MessageDetail';
2020-10-06 17:06:34 +00:00
import { ProgressModal } from './components/ProgressModal';
import { Quote } from './components/conversation/Quote';
import { StagedLinkPreview } from './components/conversation/StagedLinkPreview';
import { DisappearingTimeDialog } from './components/DisappearingTimeDialog';
import { WhatsNewLink } from './components/WhatsNewLink';
2021-07-14 23:39:52 +00:00
import { DownloadedAttachmentType } from './types/Attachment';
2021-02-18 16:40:26 +00:00
import { ElectronLocaleType } from './util/mapToSupportLocale';
import { SignalProtocolStore } from './SignalProtocolStore';
2021-03-13 01:22:36 +00:00
import { StartupQueue } from './util/StartupQueue';
2021-06-09 22:28:54 +00:00
import { SocketStatus } from './types/SocketStatus';
2021-04-08 16:24:21 +00:00
import SyncRequest from './textsecure/SyncRequest';
import { MessageController } from './util/MessageController';
import { StateType } from './state/reducer';
import { SystemTraySetting } from './types/SystemTraySetting';
import { UUID } from './types/UUID';
import { Address } from './types/Address';
import { QualifiedAddress } from './types/QualifiedAddress';
2021-08-11 19:29:07 +00:00
import { CI } from './CI';
import { IPCEventsType } from './util/createIPCEvents';
2021-08-30 21:32:56 +00:00
import { ConversationView } from './views/conversation_view';
import type { SignalContextType } from './windows/context';
import type { EmbeddedContactType } from './types/EmbeddedContact';
2020-09-04 01:25:19 +00:00
export { Long } from 'long';
2020-11-20 17:30:45 +00:00
export type WhatIsThis = any;
// Synced with the type in ts/shims/showConfirmationDialog
// we are duplicating it here because that file cannot import/export.
type ConfirmationDialogViewProps = {
cancelText?: string;
confirmStyle?: 'affirmative' | 'negative';
message: string;
okText: string;
reject?: (error: Error) => void;
resolve: () => void;
};
2021-09-29 20:23:06 +00:00
export declare class WebAudioRecorderClass {
constructor(
node: GainNode,
options: {
encoding: string;
workerDir: string;
options?: { timeLimit?: number };
}
);
// Callbacks
onComplete?: (recorder: WebAudioRecorderClass, blob: Blob) => unknown;
onError?: (recorder: WebAudioRecorderClass, error: Error) => unknown;
onTimeout?: () => unknown;
// Class properties
startRecording: () => unknown;
finishRecording: () => unknown;
isRecording: () => boolean;
cancelRecording: () => unknown;
worker: Worker;
}
declare global {
// We want to extend `window`'s properties, so we need an interface.
// eslint-disable-next-line no-restricted-syntax
interface Window {
startApp: () => void;
removeSetupMenuItems: () => unknown;
2021-11-05 08:47:32 +00:00
showPermissionsPopup: (
forCalling: boolean,
forCamera: boolean
) => Promise<void>;
FontFace: typeof FontFace;
_: typeof Underscore;
$: typeof jQuery;
moment: typeof moment;
imageToBlurHash: typeof imageToBlurHash;
loadImage: any;
isBehindProxy: () => boolean;
2021-10-27 17:54:16 +00:00
getAutoLaunch: () => Promise<boolean>;
setAutoLaunch: (value: boolean) => Promise<void>;
PQueue: typeof PQueue;
PQueueType: PQueue;
Mustache: {
render: (template: string, data: any, partials?: any) => string;
parse: (template: string) => void;
};
2021-09-29 20:23:06 +00:00
WebAudioRecorder: typeof WebAudioRecorderClass;
WhatIsThis: WhatIsThis;
2021-06-14 19:01:00 +00:00
addSetupMenuItems: () => void;
attachmentDownloadQueue: Array<MessageModel> | undefined;
2021-03-13 01:22:36 +00:00
startupProcessingQueue: StartupQueue | undefined;
baseAttachmentsPath: string;
baseStickersPath: string;
baseTempPath: string;
2022-01-11 20:02:46 +00:00
crashReports: {
getCount: () => Promise<number>;
upload: () => Promise<void>;
erase: () => Promise<void>;
};
2021-09-23 18:16:09 +00:00
drawAttention: () => void;
enterKeyboardMode: () => void;
enterMouseMode: () => void;
getAccountManager: () => AccountManager;
2021-05-03 16:38:20 +00:00
getBuiltInImages: () => Promise<Array<string>>;
getConversations: () => ConversationModelCollectionType;
getBuildCreation: () => number;
getEnvironment: typeof getEnvironment;
getExpiration: () => string;
getHostName: () => string;
getInboxCollection: () => ConversationModelCollectionType & {
onEmpty: () => void;
};
2020-10-30 17:52:21 +00:00
getInteractionMode: () => 'mouse' | 'keyboard';
2021-02-18 16:40:26 +00:00
getLocale: () => ElectronLocaleType;
2020-06-04 18:16:19 +00:00
getMediaCameraPermissions: () => Promise<boolean>;
getMediaPermissions: () => Promise<boolean>;
2020-09-09 02:25:05 +00:00
getServerPublicParams: () => string;
getSfuUrl: () => string;
2021-06-09 22:28:54 +00:00
getSocketStatus: () => SocketStatus;
getSyncRequest: (timeoutMillis?: number) => SyncRequest;
getTitle: () => string;
2020-09-09 02:25:05 +00:00
waitForEmptyEventQueue: () => Promise<void>;
getVersion: () => string;
2020-06-04 18:16:19 +00:00
i18n: LocalizerType;
isActive: () => boolean;
2021-05-03 16:38:20 +00:00
isAfterVersion: (version: string, anotherVersion: string) => boolean;
isBeforeVersion: (version: string, anotherVersion: string) => boolean;
2021-02-01 20:01:25 +00:00
isFullScreen: () => boolean;
2022-05-11 22:58:14 +00:00
initialTheme?: ThemeType;
libphonenumber: {
util: {
getRegionCodeForNumber: (number: string) => string;
parseNumber: (
e164: string,
2021-06-25 16:08:16 +00:00
defaultRegionCode?: string
) =>
| { isValidNumber: false; error: unknown }
| {
isValidNumber: true;
regionCode: string | undefined;
countryCode: string;
nationalNumber: string;
e164: string;
};
};
parse: (number: string) => string;
getRegionCodeForNumber: (number: string) => string;
format: (number: string, format: PhoneNumberFormat) => string;
};
nodeSetImmediate: typeof setImmediate;
2021-02-01 20:01:25 +00:00
onFullScreenChange: (fullScreen: boolean) => void;
2020-06-04 18:16:19 +00:00
platform: string;
preloadedImages: Array<WhatIsThis>;
reduxActions: ReduxActions;
reduxStore: Store<StateType>;
registerForActive: (handler: () => void) => void;
restart: () => void;
setImmediate: typeof setImmediate;
2020-06-04 18:16:19 +00:00
showWindow: () => void;
2020-09-16 19:31:05 +00:00
showSettings: () => void;
shutdown: () => void;
showDebugLog: () => void;
sendChallengeRequest: (request: IPCChallengeRequest) => void;
setAutoHideMenuBar: (value: WhatIsThis) => void;
setBadgeCount: (count: number) => void;
setMenuBarVisibility: (value: WhatIsThis) => void;
updateSystemTraySetting: (value: SystemTraySetting) => void;
showConfirmationDialog: (options: ConfirmationDialogViewProps) => void;
showKeyboardShortcuts: () => void;
storage: Storage;
systemTheme: WhatIsThis;
textsecure: TextSecureType;
2021-02-01 20:01:25 +00:00
titleBarDoubleClick: () => void;
unregisterForActive: (handler: () => void) => void;
updateTrayIcon: (count: number) => void;
Backbone: typeof Backbone;
2021-08-11 19:29:07 +00:00
CI?: CI;
2021-12-16 15:02:22 +00:00
2021-04-27 22:11:59 +00:00
Accessibility: {
reducedMotionSetting: boolean;
};
Signal: {
Backbone: any;
Crypto: typeof Crypto;
Curve: typeof Curve;
Data: typeof Data;
2020-10-06 17:06:34 +00:00
Groups: typeof Groups;
RemoteConfig: typeof RemoteConfig;
2020-06-04 18:16:19 +00:00
Services: {
calling: CallingClass;
enableStorageService: () => boolean;
eraseAllStorageServiceState: (options?: {
keepUnknownFields?: boolean;
}) => Promise<void>;
initializeGroupCredentialFetcher: () => void;
initializeNetworkObserver: (network: ReduxActions['network']) => void;
initializeUpdateListener: (updates: ReduxActions['updates']) => void;
2021-05-28 19:11:19 +00:00
retryPlaceholders?: Util.RetryPlaceholders;
lightSessionResetQueue?: PQueue;
runStorageServiceSyncJob: () => Promise<void>;
storageServiceUploadJob: () => void;
2020-06-04 18:16:19 +00:00
};
2020-09-09 02:25:05 +00:00
Migrations: {
2021-09-24 00:49:05 +00:00
readTempData: (path: string) => Promise<Uint8Array>;
2020-09-09 02:25:05 +00:00
deleteAttachmentData: (path: string) => Promise<void>;
2021-09-24 00:49:05 +00:00
doesAttachmentExist: (path: string) => Promise<boolean>;
writeNewAttachmentData: (data: Uint8Array) => Promise<string>;
deleteExternalMessageFiles: (attributes: unknown) => Promise<void>;
getAbsoluteAttachmentPath: (path: string) => string;
loadAttachmentData: <T extends { path?: string }>(
attachment: T
) => Promise<
T & {
data: Uint8Array;
size: number;
}
>;
loadQuoteData: (quote: unknown) => WhatIsThis;
loadContactData: (
contact?: Array<EmbeddedContactType>
) => Promise<Array<ContactWithHydratedAvatar> | undefined>;
loadPreviewData: (preview: unknown) => WhatIsThis;
loadStickerData: (sticker: unknown) => WhatIsThis;
2021-09-24 00:49:05 +00:00
readStickerData: (path: string) => Promise<Uint8Array>;
2021-07-09 19:36:10 +00:00
deleteSticker: (path: string) => Promise<void>;
getAbsoluteStickerPath: (path: string) => string;
2021-11-11 22:43:05 +00:00
processNewEphemeralSticker: (stickerData: Uint8Array) => {
2021-07-09 19:36:10 +00:00
path: string;
width: number;
height: number;
};
2021-11-11 22:43:05 +00:00
processNewSticker: (stickerData: Uint8Array) => {
2021-07-09 19:36:10 +00:00
path: string;
width: number;
height: number;
};
copyIntoAttachmentsDirectory: (
path: string
) => Promise<{ path: string; size: number }>;
upgradeMessageSchema: (attributes: unknown) => WhatIsThis;
processNewAttachment: (
2021-07-14 23:39:52 +00:00
attachment: DownloadedAttachmentType
) => Promise<DownloadedAttachmentType>;
copyIntoTempDirectory: (
path: string
) => Promise<{ path: string; size: number }>;
2021-07-09 19:36:10 +00:00
deleteDraftFile: (path: string) => Promise<void>;
deleteTempFile: (path: string) => Promise<void>;
getAbsoluteDraftPath: any;
getAbsoluteTempPath: any;
openFileInFolder: any;
2021-09-24 00:49:05 +00:00
readAttachmentData: (path: string) => Promise<Uint8Array>;
readDraftData: (path: string) => Promise<Uint8Array>;
saveAttachmentToDisk: (options: {
data: Uint8Array;
name: string;
}) => Promise<null | { fullPath: string; name: string }>;
writeNewDraftData: (data: Uint8Array) => Promise<string>;
2021-08-06 00:17:05 +00:00
deleteAvatar: (path: string) => Promise<void>;
getAbsoluteAvatarPath: (src: string) => string;
2021-09-24 00:49:05 +00:00
writeNewAvatarData: (data: Uint8Array) => Promise<string>;
2021-11-02 23:01:13 +00:00
getAbsoluteBadgeImageFilePath: (path: string) => string;
writeNewBadgeImageFileData: (data: Uint8Array) => Promise<string>;
};
2020-09-09 02:25:05 +00:00
Types: {
Message: {
CURRENT_SCHEMA_VERSION: number;
VERSION_NEEDED_FOR_DISPLAY: number;
GROUP: 'group';
PRIVATE: 'private';
initializeSchemaVersion: (version: {
message: unknown;
logger: unknown;
}) => unknown & {
schemaVersion: number;
};
hasExpiration: (json: string) => boolean;
};
Sticker: {
emoji: string;
packId: string;
packKey: string;
stickerId: number;
data: {
pending: boolean;
path: string;
};
width: number;
height: number;
path: string;
2020-09-09 02:25:05 +00:00
};
UUID: typeof UUID;
Address: typeof Address;
QualifiedAddress: typeof QualifiedAddress;
};
Util: typeof Util;
Components: {
AttachmentList: typeof AttachmentList;
2021-05-28 16:15:17 +00:00
ChatColorPicker: typeof ChatColorPicker;
ConfirmationDialog: typeof ConfirmationDialog;
ContactDetail: typeof ContactDetail;
ContactModal: typeof ContactModal;
DisappearingTimeDialog: typeof DisappearingTimeDialog;
ErrorModal: typeof ErrorModal;
Lightbox: typeof Lightbox;
MediaGallery: typeof MediaGallery;
MessageDetail: typeof MessageDetail;
2020-10-06 17:06:34 +00:00
ProgressModal: typeof ProgressModal;
Quote: typeof Quote;
StagedLinkPreview: typeof StagedLinkPreview;
WhatsNewLink: typeof WhatsNewLink;
};
OS: typeof OS;
IndexedDB: {
removeDatabase: WhatIsThis;
doesDatabaseExist: WhatIsThis;
};
Views: WhatIsThis;
2020-10-30 17:52:21 +00:00
State: {
createStore: typeof createStore;
Roots: {
2021-06-14 19:01:00 +00:00
createApp: typeof createApp;
2021-05-28 16:15:17 +00:00
createChatColorPicker: typeof createChatColorPicker;
createConversationDetails: typeof createConversationDetails;
2021-04-27 22:35:35 +00:00
createForwardMessageModal: typeof createForwardMessageModal;
createGroupLinkManagement: typeof createGroupLinkManagement;
createGroupV1MigrationModal: typeof createGroupV1MigrationModal;
createGroupV2JoinModal: typeof createGroupV2JoinModal;
createGroupV2Permissions: typeof createGroupV2Permissions;
2020-10-30 17:52:21 +00:00
createLeftPane: typeof createLeftPane;
createMessageDetail: typeof createMessageDetail;
2021-08-05 12:35:33 +00:00
createConversationNotificationsSettings: typeof createConversationNotificationsSettings;
createPendingInvites: typeof createPendingInvites;
2020-10-30 17:52:21 +00:00
createSafetyNumberViewer: typeof createSafetyNumberViewer;
createShortcutGuideModal: typeof createShortcutGuideModal;
createStickerManager: typeof createStickerManager;
createStickerPreviewModal: typeof createStickerPreviewModal;
};
Ducks: {
2021-06-14 19:01:00 +00:00
app: typeof appDuck;
2020-10-30 17:52:21 +00:00
calling: typeof callingDuck;
conversations: typeof conversationsDuck;
emojis: typeof emojisDuck;
expiration: typeof expirationDuck;
items: typeof itemsDuck;
2021-04-27 22:35:35 +00:00
linkPreviews: typeof linkPreviewsDuck;
2020-10-30 17:52:21 +00:00
network: typeof networkDuck;
updates: typeof updatesDuck;
user: typeof userDuck;
search: typeof searchDuck;
stickers: typeof stickersDuck;
};
Selectors: {
conversations: typeof conversationsSelectors;
search: typeof searchSelectors;
};
};
conversationControllerStart: WhatIsThis;
Emojis: {
getInitialState: () => WhatIsThis;
load: () => void;
};
challengeHandler: ChallengeHandler;
};
ConversationController: ConversationController;
2021-08-18 20:08:14 +00:00
Events: IPCEventsType;
MessageController: MessageController;
2021-02-26 23:42:45 +00:00
SignalProtocolStore: typeof SignalProtocolStore;
WebAPI: WebAPIConnectType;
Whisper: WhisperType;
2020-06-04 18:16:19 +00:00
getServerTrustRoot: () => WhatIsThis;
readyForUpdates: () => void;
2021-08-03 17:05:20 +00:00
logAppLoadedEvent?: (options: { processedCount?: number }) => void;
logAuthenticatedConnect?: () => void;
2020-11-13 19:57:55 +00:00
// Runtime Flags
isShowingModal?: boolean;
// Feature Flags
GV2_ENABLE_SINGLE_CHANGE_PROCESSING: boolean;
GV2_ENABLE_CHANGE_PROCESSING: boolean;
GV2_ENABLE_STATE_PROCESSING: boolean;
GV2_ENABLE_PRE_JOIN_FETCH: boolean;
GV2_MIGRATION_DISABLE_ADD: boolean;
GV2_MIGRATION_DISABLE_INVITE: boolean;
2021-08-20 16:06:15 +00:00
RETRY_DELAY: boolean;
// Context Isolation
SignalContext: SignalContextType;
}
2020-05-27 21:37:06 +00:00
// We want to extend `Error`, so we need an interface.
// eslint-disable-next-line no-restricted-syntax
2020-05-27 21:37:06 +00:00
interface Error {
2021-04-14 01:27:43 +00:00
originalError?: Event;
2021-07-09 19:36:10 +00:00
reason?: any;
stackForLog?: string;
2020-05-27 21:37:06 +00:00
}
// We want to extend `Element`'s properties, so we need an interface.
// eslint-disable-next-line no-restricted-syntax
interface Element {
// WebKit-specific
scrollIntoViewIfNeeded: (bringToCenter?: boolean) => void;
}
// Uint8Array and ArrayBuffer are type-compatible in TypeScript's covariant
// type checker, but in reality they are not. Let's assert correct use!
interface Uint8Array {
__uint8array: never;
}
interface ArrayBuffer {
__array_buffer: never;
}
interface SharedArrayBuffer {
__array_buffer: never;
}
}
export class CertificateValidatorType {
validate: (cerficate: any, certificateTime: number) => Promise<void>;
}
2020-06-04 18:16:19 +00:00
export class GumVideoCapturer {
constructor(
maxWidth: number,
maxHeight: number,
maxFramerate: number,
localPreview: Ref<HTMLVideoElement>
);
}
export class CanvasVideoRenderer {
constructor(canvas: Ref<HTMLCanvasElement>);
}
2021-08-30 21:32:56 +00:00
export class AnyViewClass extends window.Backbone.View<any> {
public headerTitle?: string;
static show(view: typeof AnyViewClass, element: Element): void;
constructor(options?: any);
}
export class BasicReactWrapperViewClass extends AnyViewClass {
public update(options: any): void;
}
export type WhisperType = {
2021-08-30 21:32:56 +00:00
Conversation: typeof ConversationModel;
ConversationCollection: typeof ConversationModelCollectionType;
Message: typeof MessageModel;
MessageCollection: typeof MessageModelCollectionType;
GroupMemberConversation: WhatIsThis;
deliveryReceiptQueue: PQueue;
deliveryReceiptBatcher: BatcherType<Receipt>;
events: Backbone.Events;
2021-08-30 21:32:56 +00:00
activeConfirmationView: WhatIsThis;
// Backbone views
// Modernized
ConversationView: typeof ConversationView;
// Note: we can no longer use 'View.extend' once we've moved to Typescript's preferred
// 'extend View' syntax. Thus, we'll need to typescriptify most of it at once.
ClearDataView: typeof AnyViewClass;
ConversationLoadingScreen: typeof AnyViewClass;
GroupMemberList: typeof AnyViewClass;
InboxView: typeof AnyViewClass;
KeyVerificationPanelView: typeof AnyViewClass;
ReactWrapperView: typeof BasicReactWrapperViewClass;
SafetyNumberChangeDialogView: typeof AnyViewClass;
View: typeof AnyViewClass;
};